OpenSCAD source code: various battery holders for blinky LED circuit, now with D alkaline cells and WWVB receiver
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Astable Multivibrator | |
// Holder for Alkaline cells | |
// Ed Nisley KE4ZNU August 2020 | |
// 2020-09 add LED radome | |
// 2020-11 add radome trim | |
// 2021-11 D cells and WWVB receiver | |
/* [Layout options] */ | |
Layout = "Build"; // [Build,Show,Lid,Spider,AntCap,RecFlag] | |
CellName = "AA"; // [AA, D] | |
Struts = -1; // [0:None, -1:Dual, 1:Quad] | |
WWVB = true; | |
/* [Hidden] */ | |
NumCells = 2; // [2] | |
// Extrusion parameters | |
/* [Hidden] */ | |
ThreadThick = 0.25; | |
ThreadWidth = 0.40; | |
HoleWindage = 0.2; | |
function IntegerMultiple(Size,Unit) = Unit * ceil(Size / Unit); | |
function IntegerLessMultiple(Size,Unit) = Unit * floor(Size / Unit); | |
Protrusion = 0.1; // make holes end cleanly | |
inch = 25.4; | |
//- Basic dimensions | |
WallThick = IntegerMultiple(3.0,ThreadWidth); | |
CornerRadius = WallThick/2; | |
FloorThick = IntegerMultiple(3.0,ThreadThick); | |
TopThick = IntegerMultiple(2.0,ThreadThick); | |
WireOD = 1.5; // battery & LED wiring | |
WireOC = 8.0; // hole spacing in lid | |
Gap = 5.0; | |
// Cylindrical cell sizes | |
// https://en.wikipedia.org/wiki/List_of_battery_sizes#Cylindrical_batteries | |
CELL_NAME = 0; | |
CELL_OD = 1; | |
CELL_OAL = 2; | |
// FIXME search() needs special-casing to properly find AAA and AAAA | |
// Which is why CellName is limited to AA | |
CellData = [ | |
["AAAA",8.3,42.5], | |
["AAA",10.5,44.5], | |
["AA",14.5,50.5], | |
["C",26.2,50], | |
["D",34.2,61.5], | |
["A23",10.3,28.5], | |
["CR123A",17.0,34.5], | |
["18650",18.8,65.2], // bare 18650 with button end | |
["18650Prot",19.0,70.0], // protected 18650 = 19670 plus a bit | |
]; | |
CellIndex = search([CellName],CellData,1,0)[0]; | |
echo(str("Cell index: ",CellIndex," = ",CellData[CellIndex][CELL_NAME])); | |
//- Contact dimensions | |
CONTACT_NAME = 0; | |
CONTACT_WIDE = 1; | |
CONTACT_HIGH = 2; | |
CONTACT_THICK = 3; // plate thickness | |
CONTACT_TIP = 4; // tip to rear face | |
CONTACT_TAB = 5; // solder tab width | |
ContactData = [ | |
["AA+",12.2,12.2,0.3,1.7,3.5], // pos bump | |
["AA-",12.2,12.2,0.3,5.0,3.5], // half-compressed neg spring | |
["AA+-",28.2,12.2,0.3,5.0,0], // pos-neg bridge | |
["D+",18.5,16.0,0.3,2.8,5.5], | |
["D-",18.5,16.0,0.3,6.0,5.5], | |
["D+-",50.0,19.0,0.3,7.0,0], // solder +/- tabs together | |
["Li+",18.5,16.0,0.3,2.8,5.5], | |
["Li-",18.5,16.0,0.3,6.0,5.5], | |
]; | |
function ConDat(name,dim) = ContactData[search([name],ContactData,1,0)[0]][dim]; | |
ContactRecess = 2*ConDat(str(CellName,"+"),CONTACT_THICK); | |
ContactOC = CellData[CellIndex][CELL_OD]; | |
WireBay = 6.0; // room for wiring to contacts | |
//- Wire struts | |
StrutDia = 1.6; // AWG 14 = 1.6 mm | |
StrutSides = 3*4; | |
ID = 0; | |
OD = 1; | |
LENGTH = 2; | |
StrutBase = [StrutDia,StrutDia + 2*5*ThreadWidth, // ID = wire, OD = buildable | |
FloorThick + CellData[CellIndex][CELL_OD]]; // LENGTH = base is flush with cell top | |
//- Holder dimensions | |
BatterySize = [CellData[CellIndex][CELL_OAL] + // cell | |
ConDat(str(CellName,"+"),CONTACT_TIP) + // pos contact | |
ConDat(str(CellName,"-"),CONTACT_TIP) - // neg contact | |
2*ContactRecess, // sink into wall | |
NumCells*CellData[CellIndex][CELL_OD], | |
CellData[CellIndex][CELL_OD] | |
]; | |
echo(str("Battery space: ",BatterySize)); | |
CaseSize = [3*WallThick + // end walls + wiring partition | |
BatterySize.x + // cell | |
WireBay, // wiring bay | |
2*WallThick + BatterySize.y, | |
FloorThick + BatterySize.z | |
]; | |
echo(str("CaseSize: ",CaseSize)); | |
BatteryOffset = (CaseSize.x - (2*WallThick + | |
CellData[CellIndex][CELL_OAL] + | |
ConDat(str(CellName,"-"),CONTACT_TIP)) | |
) /2 ; | |
ThumbRadius = 0.75 * CaseSize.z; | |
StrutOC = [IntegerLessMultiple(CaseSize.x - 2*CornerRadius -2*StrutBase[OD],5.0), | |
IntegerMultiple(CaseSize.y + StrutBase[OD],5.0)]; | |
StrutAngle = atan(StrutOC.y/StrutOC.x); | |
echo(str("Strut OC: ",StrutOC)); | |
LidSize = [2*WallThick + WireBay + ConDat(str(CellName,"+"),CONTACT_THICK), CaseSize.y, FloorThick/2]; | |
LidScrew = [2.0,3.8,7.0]; // M2 pan head screw (LENGTH = threaded) | |
LidScrewOC = CaseSize.y/2 - CornerRadius - LidScrew[OD]; // allow space around screw head | |
//- Piranha LEDs | |
PiranhaBody = [8.0,8.0,8.0]; // Z = heatsink fins + plastic body + lens | |
PiranhaPin = 0.0; // trimmed pin length beyond heatsink | |
PiranhaPinsOC = [5.0,5.0]; // pin XY distance | |
PiranhaRecess = PiranhaBody.z + PiranhaPin/2; // minimum LED recess depth | |
BallOD = 40.0; // radome sphere | |
BallSides = 4*3*4; // nice smoothness | |
PillarOD = norm([PiranhaBody.x,PiranhaBody.y]) + 2*WallThick; | |
BallChordM = BallOD/2 - sqrt(pow(BallOD/2,2) - (pow(PillarOD,2))/4); | |
echo(str("Ball chord depth: ",BallChordM)); | |
RadomePillar = [norm([PiranhaBody.x,PiranhaBody.y]), // ID = LED diagonal | |
PillarOD, | |
FloorThick + PiranhaRecess + BallChordM]; // height to top of ball chord | |
echo(str("Pillar: ",RadomePillar)); | |
RadomeBar = [StrutBase[OD]*cos(180/StrutSides),StrutOC.y,StrutBase[OD]/2]; | |
Tape = [RadomePillar[ID],16.0,1.0]; // sticky tape disk, OD to match hole punch | |
//- WWVB receiver hardware | |
Antenna = [10.0 + 0.5,14.0,60.0 + 2.0]; // ferrite antenna bar with clearance | |
AntCapSize = [Antenna[ID] + 1.0,Antenna[OD],5.0]; // LENGTH=insertion | |
RecPCB = [24.0,16.0,5.0]; | |
//---------------------- | |
// Useful routines | |
module PolyCyl(Dia,Height,ForceSides=0) { // based on nophead's polyholes | |
Sides = (ForceSides != 0) ? ForceSides : (ceil(Dia) + 2); | |
FixDia = Dia / cos(180/Sides); | |
cylinder(r=(FixDia + HoleWindage)/2,h=Height,$fn=Sides); | |
} | |
// Spider for single LED atop struts, with the ball | |
module DualSpider() { | |
difference() { | |
union() { | |
for (j=[-1,1]) { | |
for (k=[-1,1]) | |
translate([0,j*StrutOC.y/2,k*RadomeBar.z]) | |
rotate(180/StrutSides) | |
sphere(d=StrutBase[OD]/cos(180/StrutSides),$fn=StrutSides); | |
translate([0,j*StrutOC.y/2,0]) | |
rotate(180/StrutSides) | |
cylinder(d=StrutBase[OD],h=2*RadomeBar.z,center=true,$fn=StrutSides); | |
} | |
cube(RadomeBar,center=true); // connecting bar | |
cylinder(d=RadomePillar[OD],h=RadomePillar[LENGTH],$fn=BallSides); | |
translate([0,0,-RadomeBar.z/2]) | |
cylinder(d1=0.9*RadomePillar[OD],d2=RadomePillar[OD],h=RadomeBar.z/2,$fn=BallSides); | |
} | |
for (j=[-1,1]) // strut wires | |
translate([0,j*StrutOC.y/2,-3*StrutBase[OD]/2]) | |
rotate(180/StrutSides) | |
PolyCyl(StrutBase[ID],2*StrutBase[OD],StrutSides); | |
for (k=[-1,1]) // LED wiring through bar | |
translate([0,k*(StrutOC.x/2 - 2*RadomeBar.x),-RadomeBar.z]) | |
rotate(180/6) | |
PolyCyl(StrutBase[ID],2*RadomeBar.z,6); | |
translate([0,0,BallOD/2 + RadomePillar[LENGTH] - BallChordM]) // ball inset | |
sphere(d=BallOD); | |
translate([0,0,BallOD/2 + RadomePillar[LENGTH] - BallChordM - Tape[LENGTH]/2]) // tape inset | |
intersection() { | |
sphere(d=BallOD); | |
cylinder(d=Tape[OD],h=2*BallOD,center=true); | |
} | |
translate([0,0,RadomePillar.z - PiranhaRecess + RadomePillar.z/2]) // LED inset | |
cube(PiranhaBody + [HoleWindage,HoleWindage,RadomePillar.z],center=true); // XY clearance | |
translate([0,0,StrutBase[OD]/4 + WireOD/2 + 0*Protrusion]) // wire channels | |
cube([WireOD,RadomePillar[OD] + 2*WallThick,WireOD],center=true); | |
} | |
} | |
//-- WWVB antenna support cap | |
module AntennaBar() { | |
rotate([90,0,0]) | |
union() { | |
cylinder(d=Antenna[ID],h=Antenna[LENGTH],$fn=BallSides,center=true); | |
cylinder(d=2*Antenna[OD],h=Antenna[LENGTH] - 2*AntCapSize[LENGTH],$fn=BallSides,center=true); | |
} | |
} | |
module AntennaCap() { | |
rotate([90,0,0]) | |
intersection() { | |
translate([0,-Antenna[LENGTH]/2 + AntCapSize[LENGTH],0]) | |
difference() { | |
hull() { | |
rotate([90,0,0]) | |
cylinder(d=AntCapSize[OD],h=Antenna[LENGTH],$fn=BallSides,center=true); | |
for (j=[-1,1]) | |
translate([0,j*StrutOC.y/2,0]) | |
rotate(180/StrutSides) | |
cylinder(d=StrutBase[OD],h=1*StrutBase[OD],$fn=StrutSides,center=true); | |
} | |
for (j=[-1,1]) | |
translate([0,j*StrutOC.y/2,-Antenna[OD]/2]) | |
rotate(180/StrutSides) | |
PolyCyl(StrutBase[ID],Antenna[OD],StrutSides); | |
AntennaBar(); | |
} | |
rotate([-90,0,0]) | |
cylinder(d=Antenna[OD],h=Antenna[LENGTH],center=false); | |
} | |
} | |
//-- WWVB PCB support flag | |
module RecFlag() { | |
difference() { | |
hull() { | |
rotate(180/StrutSides) | |
cylinder(d=StrutBase[OD],h=RecPCB.x,$fn=StrutSides); | |
translate([0,RecPCB.y,0]) | |
rotate(180/StrutSides) | |
cylinder(d=StrutBase[OD],h=RecPCB.x,$fn=StrutSides); | |
} | |
translate([0,0,-Protrusion]) | |
rotate(180/StrutSides) | |
PolyCyl(StrutBase[ID],2*RecPCB.x,StrutSides); | |
translate([0,StrutBase[OD]/2,-Protrusion]) | |
cube([StrutBase[OD],RecPCB.y,2*RecPCB.x],center=false); | |
} | |
} | |
//-- Overall case with origin at battery center | |
module Case() { | |
union() { | |
difference() { | |
union() { | |
hull() | |
for (i=[-1,1], j=[-1,1]) | |
translate([i*(CaseSize.x/2 - CornerRadius), | |
j*(CaseSize.y/2 - CornerRadius), | |
0]) | |
cylinder(r=CornerRadius/cos(180/8),h=CaseSize.z,$fn=8); // cos() fixes undersize spheres! | |
if (Struts) | |
for (i = (Struts == 1) ? [-1,1] : -1) { // strut bases | |
hull() | |
for (j=[-1,1]) | |
translate([i*StrutOC.x/2,j*StrutOC.y/2,0]) | |
rotate(180/StrutSides) | |
cylinder(d=StrutBase[OD],h=StrutBase[LENGTH],$fn=StrutSides); | |
translate([i*StrutOC.x/2,0,StrutBase[LENGTH]/2]) | |
cube([2*StrutBase[OD],StrutOC.y,StrutBase[LENGTH]],center=true); // blocks for fairing | |
for (j=[-1,1]) // hemisphere caps | |
translate([i*StrutOC.x/2, | |
j*StrutOC.y/2, | |
StrutBase[LENGTH]]) | |
rotate(180/StrutSides) | |
sphere(d=StrutBase[OD]/cos(180/StrutSides),$fn=StrutSides); | |
} | |
} | |
translate([BatteryOffset,0,BatterySize.z/2 + FloorThick]) // cells | |
cube(BatterySize + [0,0,Protrusion],center=true); | |
translate([BatterySize.x/2 + BatteryOffset + ContactRecess/2 - Protrusion/2, // contacts | |
0, | |
BatterySize.z/2 + FloorThick]) | |
cube([ContactRecess + Protrusion, | |
ConDat(str(CellName,"+-"),CONTACT_WIDE), | |
ConDat(str(CellName,"+-"),CONTACT_HIGH) | |
],center=true); | |
translate([-(BatterySize.x/2 - BatteryOffset + ContactRecess/2 - Protrusion/2), | |
ContactOC/2, | |
BatterySize.z/2 + FloorThick]) | |
cube([ContactRecess + Protrusion, | |
ConDat(str(CellName,"+"),CONTACT_WIDE), | |
ConDat(str(CellName,"+"),CONTACT_HIGH) | |
],center=true); | |
translate([-(BatterySize.x/2 - BatteryOffset + ContactRecess/2 - Protrusion/2), | |
-ContactOC/2, | |
BatterySize.z/2 + FloorThick]) | |
cube([ContactRecess + Protrusion, | |
ConDat(str(CellName,"-"),CONTACT_WIDE), | |
ConDat(str(CellName,"-"),CONTACT_HIGH) | |
],center=true); | |
translate([-CaseSize.x/2 + WireBay/2 + WallThick, // wire bay with screw bosses | |
0, | |
BatterySize.z/2 + FloorThick + Protrusion/2]) | |
cube([WireBay, | |
2*LidScrewOC - LidScrew[ID] - 2*4*ThreadWidth, | |
BatterySize.z + Protrusion | |
],center=true); | |
for (j=[-1,1]) // screw holes | |
translate([-CaseSize.x/2 + WireBay/2 + WallThick, | |
j*LidScrewOC, | |
CaseSize.z - LidScrew[LENGTH] + Protrusion]) | |
PolyCyl(LidScrew[ID],LidScrew[LENGTH],6); | |
for (j=[-1,1]) | |
translate([-(BatterySize.x/2 - BatteryOffset + WallThick/2), // contact tabs | |
j*ContactOC/2, | |
BatterySize.z + FloorThick - Protrusion]) | |
cube([2*WallThick, | |
ConDat(str(CellName,"+"),CONTACT_TAB), | |
(BatterySize.z - ConDat(str(CellName,"+"),CONTACT_HIGH)) | |
],center=true); | |
if (false) | |
translate([0,0,CaseSize.z]) // finger cutout | |
rotate([90,00,0]) | |
cylinder(r=ThumbRadius,h=2*CaseSize.y,center=true,$fn=22); | |
if (Struts) | |
for (i2 = (Struts == 1) ? [-1,1] : -1) { // strut wire holes and fairing | |
for (j=[-1,1]) | |
translate([i2*StrutOC.x/2,j*StrutOC.y/2,FloorThick]) | |
rotate(180/StrutSides) | |
PolyCyl(StrutBase[ID],2*StrutBase[LENGTH],StrutSides); | |
for (i=[-1,1], j=[-1,1]) // fairing cutaways | |
translate([i*StrutBase[OD] + (i2*StrutOC.x/2), | |
j*StrutOC.y/2, | |
-Protrusion]) | |
rotate(180/StrutSides) | |
PolyCyl(StrutBase[OD],StrutBase[LENGTH] + 2*Protrusion,StrutSides); | |
} | |
translate([0,0,ThreadThick - Protrusion]) // recess around name | |
cube([51.0,15,2*ThreadThick],center=true); | |
} | |
linear_extrude(height=2*ThreadThick + Protrusion,convexity=10) { | |
translate([0,-3.5,0]) | |
mirror([0,1,0]) | |
text(text="softsolder",size=6,spacing=1.20,font="Arial:style:Bold",halign="center",valign="center"); | |
translate([0,3.5,0]) | |
mirror([0,1,0]) | |
text(text=".com",size=6,spacing=1.20,font="Arial:style:Bold",halign="center",valign="center"); | |
} | |
} | |
} | |
module Lid() { | |
difference() { | |
hull() | |
for (i=[-1,1], j=[-1,1], k=[-1,1]) | |
translate([i*(LidSize.x/2 - CornerRadius), | |
j*(LidSize.y/2 - CornerRadius), | |
k*(LidSize.z - CornerRadius)]) // double thickness for flat bottom | |
sphere(r=CornerRadius/cos(180/8),$fn=8); | |
translate([0,0,-LidSize.z]) // remove bottom | |
cube([(LidSize.x + 2*Protrusion),(LidSize.y + 2*Protrusion),2*LidSize.z],center=true); | |
for (j=[-1,1]) // wire holes | |
translate([0,j*WireOC/2,-Protrusion]) | |
PolyCyl(WireOD,2*LidSize.z,6); | |
for (j=[-1,1]) | |
translate([0,j*LidScrewOC,-Protrusion]) | |
PolyCyl(LidScrew[ID],2*LidSize.z,6); | |
} | |
} | |
//------------------- | |
// Show & build stuff | |
if (Layout == "Case") | |
Case(); | |
if (Layout == "Lid") | |
Lid(); | |
if (Layout == "AntCap") | |
AntennaCap(); | |
if (Layout == "RecFlag") | |
RecFlag(); | |
if (Layout == "Spider") | |
if (Struts == -1) | |
DualSpider(); | |
else | |
cube(10,center=true); | |
if (Layout == "Build") { | |
rotate(90) | |
Case(); | |
translate([0,-(CaseSize.x/2 + LidSize.x/2 + Gap),0]) | |
rotate(90) | |
Lid(); | |
if (Struts == -1) { | |
difference() { | |
union() { | |
translate([CaseSize.x/2 + RadomePillar[OD],0,0]) | |
DualSpider(); | |
translate([-(CaseSize.x/2 + RadomePillar[OD]),0,0]) | |
rotate([180,0,0]) | |
DualSpider(); | |
} | |
translate([0,0,-2*CaseSize.z]) | |
rotate(90) | |
cube(4*CaseSize,center=true); | |
} | |
} | |
if (WWVB) { | |
for (i=[-1,1]) | |
translate([i*(Antenna[LENGTH]/2 - AntCapSize[LENGTH]),CaseSize.x/2 + Antenna[OD],0]) | |
AntennaCap(); | |
translate([0,CaseSize.x/2 + Antenna[OD],0]) | |
RecFlag(); | |
} | |
} | |
if (Layout == "Show") { | |
Case(); | |
for (j=[-1,1]) | |
color("Brown",0.3) | |
translate([-StrutOC.x/2,j*StrutOC.y/2,Protrusion]) | |
cylinder(d=StrutDia[ID],h=3*CaseSize.z,$fn=StrutSides); | |
translate([-(CaseSize.x/2 - LidSize.x/2),0,(CaseSize.z + Gap)]) | |
Lid(); | |
if (Struts == -1) | |
translate([-StrutOC.x/2,0,3*CaseSize.z]) | |
DualSpider(); | |
if (WWVB) { | |
for (j=[-1,1]) | |
translate([-StrutOC.x/2,,j*(Antenna[LENGTH]/2 - AntCapSize[LENGTH]),1.5*CaseSize.z]) | |
rotate([-j*90,0,0]) | |
AntennaCap(); | |
translate([-StrutOC.x/2,,-(StrutOC.y/2),2*CaseSize.z]) | |
RecFlag(); | |
} | |
} | |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
More details on my blog at https://wp.me/poZKh-avT