1
0
mirror of https://github.com/nophead/NopSCADlib.git synced 2025-01-17 13:38:16 +01:00
NopSCADlib/vitamins/radial.scad
Chris Palmer 2d091b9d16 Added a constructor function for PCBs.
PCB text can now specify a colour.
Added SOT23_6, TSOT23_8 and CAP1210 SMD packages.
Added ESP-201 Wifi Module PCB.
Added ESP12F Wifi module.
Added tiny_buck converter PCB.
Made 2p54joiner a bit longer.
Fixed SOIC gullwing shape.
Added Wifi antennas.
Added Yunpen filtered IEC mains inlet.
Added power jack socket.
Added right angle option for LEDs.
Added gear_motors.
Added 686ZZ and 696ZZ ball bearings.
Added two larger box section tubes.
Added right angle option to square_buttons. Doesn't draw the frame yet.
Added multiwatt11 package for L6203.
Added ONS9143A 13A mains socket.
Added radial electrolytic capacitors.
Added LDE10_20B PSU module.
Added screw_tearsink() to make horizontal countersunk holes.
Can now have solid tracks on veroboard.
Added veroboard_base() module to make an STL for a base with spacers.
Corrected M6 spring washer thickness.
Can now specify the height of a solder meniscus.
slot() now has a center option.
Added button_6mm_7 with a taller button.
Added default fa, fs and fn constants used for drawing vitamins.
These can be set via $default_fa and $default_fs that can also be set
by environment variables: NOPSCADLIB_DEFAULT_FA and NOPSCADLIB_DEFAULT_FS.
2023-10-29 21:56:08 +00:00

396 lines
14 KiB
OpenSCAD

//
// NopSCADlib Copyright Chris Palmer 2023
// nop.head@gmail.com
// hydraraptor.blogspot.com
//
// This file is part of NopSCADlib.
//
// NopSCADlib is free software: you can redistribute it and/or modify it under the terms of the
// GNU General Public License as published by the Free Software Foundation, either version 3 of
// the License, or (at your option) any later version.
//
// NopSCADlib is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
// without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
// See the GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License along with NopSCADlib.
// If not, see <https://www.gnu.org/licenses/>.
//
//
//! Radial components for PCBs.
//
include <../utils/core/core.scad>
use <../utils/sweep.scad>
use <../utils/rounded_polygon.scad>
use <../utils/rounded_cylinder.scad>
use <../utils/round.scad>
use <../utils/pcb_utils.scad>
use <../utils/bezier.scad>
function rd_xtal_size(type) = type[1]; //! Crystal length, width and height and optional corner radius
function rd_xtal_flange(type) = type[2]; //! Crystal flange width and thickness
function rd_xtal_pitch(type) = type[3]; //! Crystal lead pitch
function rd_xtal_lead_d(type) = type[4]; //! Crystal lead diameter
module lead_positions(p, z) {
if(is_list(p))
for($x = [-1, 1], $y = [-1, 1])
translate([$x * p.x / 2, $y * p.y / 2, z])
children();
else
for($x = [-1, 1])
translate([$x * p / 2, 0, z])
children();
}
module radial_lead(start, end, z, tail, lead) {
$fn = fn;
profile = is_list(lead) ? rectangle_points(lead.x , lead.y) : circle_points(lead / 2);
color(silver)
if(start == end)
translate([start.x, start.y, -tail])
linear_extrude(tail + z)
polygon([for(p = profile) [p.x, p.y]]);
else {
dz = 2 * [0, 0, is_list(lead) ? norm(lead) : lead];
top = [start.x, start.y, z];
bot = [end.x, end.y, 0];
path = [top, top - dz, bot + dz, bot];
rpath = concat(bezier_path(path, 20), [bot - [0, 0, tail]]);
sweep(rpath, profile);
}
translate(end)
solder((is_list(lead) ? min(lead) : lead) / 2);
}
module radial_leads(ap, p, z, d, tail)
color(silver) {
$fn = fn;
lead_positions(p, 0)
solder(d / 2);
if(p != ap) {
assert(!is_list(p), "Bending four leads not supported yet");
dz = d;
dx = p / 2 - ap / 2;
path = [[0, z, 0], [0, z - dz, 0], [dx, dz, 0], [dx, 0, 0]];
rpath = concat(bezier_path(path, 20), [[dx, -tail, 0]]);
lead_positions(ap, 0)
rotate([90, 0, 90 * -$x + 90])
sweep(rpath, circle_points(d / 2));
}
else
lead_positions(p, -tail)
rotate(90)
cylinder(d = d, h = tail + z);
}
module rd_xtal(type, value, z = 0, pitch = undef, tail = 3) { //! Draw a crystal
vitamin(str("rd_xtal(", type[0], ", \"", value, "\"): Crystal ", type[0], " ", value));
s = rd_xtal_size(type);
r = len(s) < 4 ? s.y / 2 - eps : s[3];
f = rd_xtal_flange(type);
cp = rd_xtal_pitch(type);
p = is_undef(pitch) ? cp : pitch;
d = rd_xtal_lead_d(type);
r2 = 0.2;
$fs = fs; $fa = fa;
color(silver) {
translate_z(z) {
if(s.y) {
rounded_rectangle([s.x, s.y, s.z - r2], r);
translate_z(s.z - r2)
rounded_top_rectangle([s.x, s.y, r2], r, r2);
}
else
rounded_cylinder(r = s.x / 2, h = s.z, r2 = r2);
if(f) {
rounded_rectangle([s.x + 2 * f.x, s.y + 2 * f.x, f[1]], r + f.x);
if(is_list(cp))
translate([-s.x / 2 - f.x, -s.y / 2 - f.x])
cube([r + f.x, r + f.x, f[1]]); // Pin 1 marked by sharp corner on 4 pin packages
}
}
radial_leads(cp, p, z, d, tail);
}
color(grey(10)) {
if(!is_undef(value))
if(s.y)
translate_z(z + s.z)
linear_extrude(eps)
resize([s.x * 0.75, s.y / 2])
text(value, halign = "center", valign = "center");
else
translate_z(z + s.z / 2)
let($fn = 32)
cylindrical_wrap(s.x / 2)
rotate(-90)
resize([s.z * 0.9, s.x * PI / 4])
text(value, halign = "center", valign = "center");
if(s.y)
lead_positions(cp, z)
cylinder(d = d * 4, h = 2 * eps, center = true);
else
translate_z(z)
cylinder(d = (s.x + cp) / 2, h = 2 * eps, center = true);
}
}
function rd_module_kind(type) = type[1]; //! Relay, PSU, etc.
function rd_module_size(type) = type[2]; //! Size
function rd_module_radius(type) = type[3]; //! Corner radius
function rd_module_colour(type) = type[4]; //! Colour
function rd_module_pin_size(type) = type[5]; //! Pin size
function rd_module_pin_posns(type) = type[6]; //! list of pin positions
module rd_module(type, value) { //! Draw a PCB mounted potted module, e.g. PSU or relay
vitamin(str("rd_module(", type[0], ", \"", value, "\"): ", rd_module_kind(type), " ", type[0], " / ", value));
$fs = fs; $fa = fa;
r = rd_module_radius(type);
size = rd_module_size(type);
pin = rd_module_pin_size(type);
color(rd_module_colour(type))
rounded_top_rectangle(size, r, r);
color(silver)
for(pos = rd_module_pin_posns(type))
translate(pos) {
translate_z(-pin.z / 2)
if(pin.y)
cube(pin, center = true);
else
cylinder(d = pin.x, h = pin.z, center = true);
solder();
}
color("white")
translate([0, -size.y / 2])
rotate([90, 0, 0])
linear_extrude(eps) {
translate([0, size.z * 0.9])
resize([size.x * 0.5, size.z / 9])
text(type[0], halign = "center", valign = "top");
translate([-size.x * 0.45, size.z * 0.75])
resize([size.x * 0.4, size.z / 12])
text(value, halign = "left", valign = "top");
}
}
function rd_disc_kind(type) = type[1]; //! Capacitor, etc
function rd_disc_size(type) = type[2]; //! Diameter, thickness and height
function rd_disc_pitch(type) = type[3]; //! Lead pitch X & Y
function rd_disc_lead_d(type) = type[4]; //! Lead diameter and sleeve diameter
function rd_disc_colours(type) = type[5]; //! Colours of body and text
module rd_disc(type, value, pitch = undef, z = 0, tail = 3) { //! Draw a radial disc component
vitamin(str("rd_disc(", type[0], ", \"", value, "\"): ", rd_disc_kind(type), ", ", type[0], " ", value));
size = rd_disc_size(type);
colours = rd_disc_colours(type);
opitch = rd_disc_pitch(type);
pitch = is_undef(pitch) ? opitch : pitch;
lead_d = rd_disc_lead_d(type);
$fs = fs; $fa = fa;
lead_positions = [for(side = [-1,1]) [-side * opitch.x / 2, side * opitch.y / 2]];
r = size / 2;
v = [[0, r.y], [r.x, r.y], [r.x, r.y * pow((r.y / r.x), 4)], [r.x, 0]];
bez = bezier_path(v, 20);
path = concat(bez, [for(p = reverse(bez)) [p.x, - p.y]]);
rotate(is_list(opitch) ? atan2(opitch.y, opitch.x): 0) {
color(colours[0]) {
translate_z(size.z - size.x / 2 + z)
rotate([90, 0, 0])
color(colours[0])
rotate_extrude()
polygon(path);
r = lead_d[1] / 2;
rl = lead_d[0] / 2;
h = size.z - size.x / 2;
for(p = lead_positions)
translate([p.x, p.y, z + r]) {
dy = (size.y / 2 - r - 0.1) * sign(-p.x);
path = [[0, 0, 0],
[0, 0, h / 2],
[-p.x / 2, dy - p.y, h / 2],
[-p.x, dy - p.y, h]];
sweep(concat([[0, 0, - r / 2]],bezier_path(path, 20)), circle_points(r));
vflip()
rounded_cylinder(r = r, h = r, r2 = r - rl, ir = rl);
}
}
diagonal_pitch = norm(opitch);
pitch = is_undef(pitch)? diagonal_pitch : pitch;
rotate(is_list(opitch) ? -atan2(opitch.y, opitch.x): 0)
radial_leads(diagonal_pitch, pitch, z, lead_d[0], tail);
}
}
function rd_transistor_size(type) = type[1]; //! Width / diameter, depth / flat and height
function rd_transistor_colours(type) = type[2]; //! Body colour and text colour
function rd_transistor_lead(type) = type[3]; //! Lead diameter or width and depth
function rd_transistor_lead_posns(type) = type[4]; //! List of lead xy coordinates
module rd_transistor(type, value, kind = "Transistor", lead_positions = undef, z = 5, tail = 3) { //! Draw a radial lead transistor
vitamin(str("rd_transistor(", type[0], ", \"", value, "\"): ", kind, " ", type[0], " ", value));
size = rd_transistor_size(type);
colours = rd_transistor_colours(type);
$fs = fs; $fa = fa;
translate_z(z) {
if(type[0] == "TO92") {
color(colours[0])
linear_extrude(size.z)
difference() {
circle(d = size.z);
translate([0, size.x / 2])
square([size.x + 1, 2 * (size.x - size.y)], center = true);
}
color(colours[1])
translate([0, -size.x / 2 + size.y, size.z / 2])
rotate([0, 90, 90])
linear_extrude(eps)
resize([size.z * 0.8, 0], auto = true)
text(value, valign = "center", halign = "center");
}
if(type[0] == "E_LINE") {
color(colours[0])
linear_extrude(size.z)
hull() {
for(side = [-1, 1])
translate([side * (size.x - size.y) / 2, 0])
circle(d = size.y);
translate([-size.x / 2, 0])
square([size.x, size.y / 2]);
}
color(colours[1])
translate([0, size.y / 2, size.z / 2])
rotate([-90, 180, 0])
linear_extrude(eps)
resize([size.x * 0.85, 0], auto = true)
text(value, valign = "center", halign = "center");
}
}
lead_positions = is_undef(lead_positions) ? [for(i = [-1:1]) [inch(0.1 * i), 0]] : lead_positions;
lead_starts = rd_transistor_lead_posns(type);
lead = rd_transistor_lead(type);
assert(len(lead_positions) == len(lead_starts), "must give a position for each lead");
for(i = [0 : len(lead_starts) - 1]) {
start = lead_starts[i];
end = lead_positions[i];
radial_lead(start, end, z, tail, lead);
}
}
function rd_electrolytic_size(type) = type[1]; //! Diameter, crimp diameter, height
function rd_electrolytic_radius(type) = type[2]; //! Corner radius
function rd_electrolytic_crimp(type) = type[3]; //! Width and height of crimp
function rd_electrolytic_lead_d(type) = type[4]; //! Lead diameter
function rd_electrolytic_pitch(type) = type[5]; //! Lead pitch
function rd_electrolytic_colours(type) = type[6]; //! Colour of jacket and stripe
module rd_electrolytic(type, value, pitch = undef, z = 0, tail = 3) { //! Draw a radial electrolytic capcacitor
vitamin(str("rd_electrolytic(", type[0], ", \"", value, "\"): Electolytic capacitor ", type[0], " ", value));
size = rd_electrolytic_size(type);
crimp = rd_electrolytic_crimp(type);
colours = rd_electrolytic_colours(type);
opitch = rd_electrolytic_pitch(type);
pitch = is_undef(pitch) ? opitch : pitch;
lead_d = rd_electrolytic_lead_d(type);
jacket_t = 0.2;
jacket_ir = size.x * 5 / 16;
cross = 0.2 / sqrt(2);
$fs = fs; $fa = fa;
radial_leads(opitch, pitch, z + jacket_t, lead_d, tail);
x = (size.x - size.y) / 2;
h = crimp[0] / 2;
r = (h / x - x) / 2;
stripe_angle = 50;
module profile()
difference() {
round(rd_electrolytic_radius(type))
difference() {
square([size.x / 2, size.z]);
translate([size.x / 2 - x + r, crimp.y])
circle(r);
}
square([jacket_ir, size.z]);
}
translate_z(z) {
color(colours[0])
rotate(stripe_angle / 2)
rotate_extrude(angle = 360 - stripe_angle)
profile();
color(colours[1]) {
rotate(-stripe_angle / 2)
rotate_extrude(angle = stripe_angle)
profile();
z0 = crimp.y + crimp.x / 2;
translate_z((size.z + z0) / 2)
cylindrical_wrap(size.x / 2)
rotate(-90)
resize([(size.z - z0) * 0.9, 0], auto = true)
text(value, halign = "center", valign = "center");
}
color(silver)
translate_z(size.z - 1)
render() difference() {
cylinder(r = jacket_ir, h = 1 - jacket_t);
translate_z(1 - jacket_t)
for(a = [0, 90])
rotate([45, 0, a])
cube([size.x, cross, cross], center = true);
}
color(grey(30))
translate_z(jacket_t)
cylinder(r = jacket_ir, h = eps);
}
}