BOSL2/gears.scad
2024-05-20 19:42:07 -04:00

4579 lines
228 KiB
OpenSCAD
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

//////////////////////////////////////////////////////////////////////////////////////////////
// LibFile: gears.scad
// Spur Gears, Bevel Gears, Racks, Worms and Worm Gears.
// Inspired by code by Leemon Baird, 2011, Leemon@Leemon.com
// Includes:
// include <BOSL2/std.scad>
// include <BOSL2/gears.scad>
// FileGroup: Parts
// FileSummary: Gears, racks, worms, and worm gears.
//////////////////////////////////////////////////////////////////////////////////////////////
_GEAR_PITCH = 5;
_GEAR_HELICAL = 0;
_GEAR_THICKNESS = 10;
_GEAR_PA = 20;
$parent_gear_type = undef;
$parent_gear_pitch = undef;
$parent_gear_teeth = undef;
$parent_gear_pa = undef;
$parent_gear_helical = undef;
$parent_gear_thickness = undef;
$parent_gear_dir = undef;
$parent_gear_travel = 0;
function _inherit_gear_param(name, val, pval, dflt, invert=false) =
is_undef(val)
? is_undef(pval)
? dflt
: (invert?-1:1)*pval
: is_undef(pval)
? assert(is_finite(val), str("Invalid ",name," value: ",val))
val
: (invert?-1:1)*val;
function _inherit_gear_pitch(fname,pitch,circ_pitch,diam_pitch,mod,warn=true) =
pitch != undef?
assert(is_finite(pitch) && pitch>0)
warn? echo(str(
"WARNING: The use of the argument pitch= in ", fname,
" is deprecated. Please use circ_pitch= instead."
)) pitch : pitch :
circ_pitch != undef?
assert(is_finite(circ_pitch) && circ_pitch>0)
circ_pitch :
diam_pitch != undef?
assert(is_finite(diam_pitch) && diam_pitch>0)
circular_pitch(diam_pitch=diam_pitch) :
mod != undef?
assert(is_finite(mod) && mod>0)
circular_pitch(mod=mod) :
$parent_gear_pitch != undef? $parent_gear_pitch :
5;
function _inherit_gear_pa(pressure_angle) =
_inherit_gear_param("pressure_angle", pressure_angle, $parent_gear_pa, dflt=20);
function _inherit_gear_helical(helical,invert=false) =
_inherit_gear_param("helical", helical, $parent_gear_helical, dflt=0, invert=invert);
function _inherit_gear_thickness(thickness,dflt=10) =
_inherit_gear_param("thickness", thickness, $parent_gear_thickness, dflt=dflt);
// Section: Quick Introduction to Gears
// This section gives a quick overview of gears with a focus on the information you need
// to know to understand the gear parameters and create some gears. The topic of gears is very complex and highly technical and
// this section provides the minimal information needed for gear making. If you want more information about the
// details of gears, consult the references below, which are the ones that we consulted when writing the library code.
// - Tec Science
// * [Involute Gears](https://www.tec-science.com/mechanical-power-transmission/involute-gear/geometry-of-involute-gears/)
// * [Gear engagement](https://www.tec-science.com/mechanical-power-transmission/involute-gear/meshing-line-action-contact-pitch-circle-law/)
// * [Gears meshing with racks](https://www.tec-science.com/mechanical-power-transmission/involute-gear/rack-meshing/)
// * [Gear undercutting](https://www.tec-science.com/mechanical-power-transmission/involute-gear/undercut/)
// * [Profile shifting](https://www.tec-science.com/mechanical-power-transmission/involute-gear/profile-shift/)
// * [Detailed gear calculations](https://www.tec-science.com/mechanical-power-transmission/involute-gear/calculation-of-involute-gears/)
// * [Worm drive](https://www.tec-science.com/mechanical-power-transmission/gear-types/worms-and-worm-gears/)
// * [Bevel gears](https://www.tec-science.com/mechanical-power-transmission/gear-types/bevel-gears/)
// - SDPSI (A long document covering a variety of gear types and gear calculations)
// * [Elements of Gear Technology](https://www.sdp-si.com/resources/elements-of-metric-gear-technology/index.php)
// - Drivetrain Hub (A collection of "notebooks" on some gear topics)
// * [Gear Geometry, Strength, Tooling and Mechanics](https://drivetrainhub.com/notebooks/#toc)
// - Crown Face Gears
// * [Crown Gearboxes](https://mag.ebmpapst.com/en/industries/drives/crown-gearboxes-efficiency-energy-savings-decentralized-drive-technology_14834/)
// * [Crown gear pressure angle](https://mag.ebmpapst.com/en/industries/drives/the-formula-for-the-pressure-angle_14624/)
// * [Face Gears: Geometry and Strength](https://www.geartechnology.com/ext/resources/issues/0107x/kissling.pdf)
// Subsection: Involute Spur Gears
// The simplest gear form is the involute spur gear, which is an extrusion of a two dimensional form.
// Figure(3D,Med,NoAxes,VPT=[4.62654,-1.10349,0.281802],VPR=[55,0,25],VPD=236.957): Involute Spur Gear
// spur_gear(mod=5,teeth=18,pressure_angle=20,thickness=25,shaft_diam=15);
// Continues:
// The term "involute" refers to the shape of the teeth: the curves of the teeth are involutes of circles,
// which are curves that optimize gear performance.
// Figure(2D,Med,NoAxes,VPT=[8,74,0],VPR=[0,0,0],VPD=150): The three marked circles are key references on gear teeth. The pitch circle, which is roughly in the middle of the teeth, is the reference used to define the pitch of teeth on the gear. The pressure angle is the angle the tooth makes with the pitch circle. In this example, the pressure angle is 20 degrees as shown by the red lines.
// $fn=128;
// intersection(){
// spur_gear2d(mod=5,teeth=30,pressure_angle=20);
// back(82)rect([45, 20],anchor=BACK);
// }
// color("black"){
// stroke(arc(r=_root_radius(mod=5,teeth=30),angle=[70,110]),width=.25);
// stroke(arc(r=pitch_radius(mod=5,teeth=30),angle=[70,110]),width=.25);
// stroke(arc(r=outer_radius(mod=5,teeth=30),angle=[70,110]),width=.25);
// back(63.5)right(24.2)text("root circle",size=2.5);
// back(69.5)right(26.5)text("pitch circle",size=2.5);
// back(74)right(28)text("outer circle",size=2.5);
// }
// base = _base_radius(mod=5, teeth=30);
// pitchpt = pitch_radius(mod=5, teeth=30);
// color("red"){
// zrot(87-360/30) zrot(20,cp=[pitchpt,0]) stroke([[base-5,0],[base+15,0]], width=0.25);
// zrot(87-360/30) stroke([[pitchpt,0],[pitchpt+11,0]], width=0.25);
// right(8.3) back(74) zrot(87-360/30) zrot(10,cp=[pitchpt,0]) stroke(arc(angle=[0,20],r=10.5),endcaps="arrow2",width=.25);
// back(84) right(13) text("pressure angle",size=2.5);
// }
// stroke(arc(r=pitch_radius(mod=5,teeth=30),angle=[87,87+12]),width=.4,endcaps="arrow2",color="red");
// color([1,0,0,1]) back(70)right(-13)zrot(4)text("circular pitch", size=2.5);
// Continues:
// The size of the teeth can be specified as the *circular pitch*, which is the tooth width, or more precisely,
// the distance along the pitch circle from the start of one tooth to the start of the text tooth.
// The circular pitch can be computed as
// `PI*d/teeth` where `d` is the diameter of the pitch circle and `teeth` is the number of teeth on the gear.
// This simply divides up the pitch circle into the specified number of teeth. However, the customary
// way to specify metric gears is using the module, ratio of the diameter of the gear to the number of teeth: `m=d/teeth`.
// The module is hence the circular pitch divided by a factor of π. A third way to specify gear sizes is the diametral pitch,
// which is the number of teeth that fit on a gear with a diameter of one inch, or π times the number of teeth per inch.
// Note that for the module or circular pitch, larger values make larger teeth,
// but for the diametral pitch, the opposite is true. Throughout this library, module and circular pitch
// are specified basic OpenSCAD units, so if you work in millimeters and want to give circular pitch in inches, be
// sure to multiply by `INCH`. The diametral pitch is given based on inches under the assumption that OpenSCAD units are millimeters.
// .
// Note that there is no direct way to specify the size of a gear. The diameter of a gear depends on its tooth count
// and tooth size. If you want a gear with a particular diameter you can get close by seeting the module to `d/teeth`,
// but that specifies the pitch circle, so the gear teeth will have a somewhat larger radius. You should **not**
// apply scale() to gears. Always change their size by adjusting the tooth size parameters.
// .
// Basic gears as shown above will mesh when their pitch circles are tangent.
// The critical requirements for two gears to mesh are that
// - The teeth are the same size
// - The pressure angles are identical
// .
// Increasing pressure angle makes the tooth stronger, increases power transmission, and can reduce tooth interference for
// gears with a small number of teeth, but it also increases gear wear and meshing noise. Higher pressure angles also
// increase the force that tries to push the gears apart, and hence the load on the gear axles. The current standard pressure
// angle is 20 degrees. It replaces an old 14.5 degree standard.
// Figure(2D,Med,NoAxes): Teeth of the same size with different pressure angles. Note that 20 deg is the industry standard.
// pang = [30,20,14.5];
// ycopies(n=3, spacing=25){
// intersection(){
// spur_gear2d(mod=5, teeth=30, pressure_angle=pang[$idx]);
// back(82) rect([45,20], anchor=BACK);
// }
// back(68) right(26) text(str(pang[$idx]), size=6.5);
// }
// Continues:
// In order for the gear teeth to fit together, and to allow space for lubricant, the valleys of the teeth
// are made deeper by the `clearance` distance. This defaults to `module/4`.
// Figure(2D,Med,NoAxes,VPT=[5.62512,-1.33268,-0.0144912],VPR=[0,0,0],VPD=126): The clearance is extra space at the tooth valley that separates the tooth tip (in green) from the tooth valley below it.
// intersection(){
// rack2d(mod=5, teeth=10, bottom=15, pressure_angle=14.5);
// rect([35,20]);
// }
// color("lightgreen")render()
// intersection(){
// back(gear_dist(mod=5, teeth1=146, teeth2=0 ,profile_shift1=0))
// spur_gear2d(mod=5, teeth=146, profile_shift=0, pressure_angle=14.5);
// rect([45,20]);
// }
// color("black") {
// stroke([[-10,-5],[20,-5]], width=.25);
// stroke([[-10,-6.2],[20,-6.2]], width=.25);
// fwd(6.4) right(22) text("clearance", size=2.5);
// }
// Continues:
// Another clearance requirement can present a serious problem when the number of teeth is low. As the gear rotates, the
// teeth may interfere with each other. This may require undercutting the gear teeth to create space, which weakens the teeth.
// Is is best to avoid gears with very small numbers of teeth when possible.
// Figure(2D,Med,NoAxes,VPT=[0.042845,6.5338,-0.0144912],VPR=[0,0,0],VPD=126): The green gear with only five teeth has a severe undercut, which weakens its teeth. This undercut is necessary to avoid interference with the teeth from the other gear during rotation. Note that the yellow rack tooth is deep into the undercut space.
// ang=16;
// rack2d(mod=5, teeth=3, bottom=15, pressure_angle=14.5, rounding=0);
// left(2*PI*pitch_radius(mod=5, teeth=5)*ang/360)
// color("lightgreen")
// back(gear_dist(mod=5, teeth1=5, profile_shift1=0, teeth2=0))
// zrot(ang)
// spur_gear2d(mod=5, teeth=5, clearance=.00001, profile_shift=0, pressure_angle=14.5, shaft_diam=5);
// Subsection: Corrected Gears and Profile Shifting
// A solution to the problem of undercutting is to use profile shifting. Profile shifting uses a different portion of the
// involute curve to form the gear teeth, and this adjustment to the tooth form can eliminate undercutting, while
// still allowing the gear to mesh with unmodified gears. Profile shifting
// changes the diameter at which the gear meshes so it no longer meshes at the pitch circle.
// A profile shift of `x`
// will increase the mesh distance by approximately `x*m` where `m` is the gear module. The exact adjustment,
// which you compute with {{gear_dist()}}, is a complex calculation that depends on the profile shifts of both meshing gears. This means that profile shifting
// can also be used to fine tune the spacing between gears. When the gear has many teeth a negative profile shift may
// be able to bring the gears slightly closer together, while still avoiding undercutting.
// Profile shifting also changes the effective pressure angle of the gear engagement.
// Figure(2D,Med,NoAxes): The green gear is a 7 tooth gear without profile shifting. In yellow is the same gear, profile shifted. Note that the teeth too longer narrow at their base. Also note that the effective root circle has a larger radius, and the teeth are also longer.
// spur_gear2d(mod=5, teeth=7);
// color("green")spur_gear2d(mod=5, teeth=7, profile_shift=0);
// Continues:
// The minimum number of teeth to avoid undercutting is 17 for a pressure angle of 20, but it is 32 for a pressure
// angle of 14.5 degrees. It can be computed as `2/(sin(alpha))^2` where `alpha` is the pressure angle.
// By default, the gear modules produce corrected gears. You can override this by specifying the profile shift
// yourself. A small undercut may be acceptable, for example: a rule of thumb indicates that gears as small as 14
// teeth are OK with a 20 degree pressure angle, because the undercut is too small to weaken the teeth significantly.
// Figure(2D,Med,NoAxes,VPT=[1.33179,10.6532,-0.0144912],VPR=[0,0,0],VPD=155.556): Basic five tooth gear form on the left. Corrected gear with profile shifting on the right. The profile shifted teeth lack the weak undercut section. The axis of the corrected gear is shifted away from the mating rack.
// $fn=32;
// ang1=-20;
// ang2=20;
// color("blue")
// left(2*PI*pitch_radius(mod=5, teeth=5)*ang1/360)
// left(3*5*PI/2)
// back(gear_dist(mod=5,teeth1=5,profile_shift1=0,teeth2=0,pressure_angle=14.5))
// zrot(ang1)
// spur_gear2d(mod=5, teeth=5, profile_shift=0, pressure_angle=14.5, shaft_diam=2);
// color("green")
// left(2*PI*pitch_radius(mod=5, teeth=5)*ang2/360)
// right(3*5*PI/2)
// back(gear_dist(mod=5, teeth1=5, teeth2=0,pressure_angle=14.5))
// zrot(ang2)
// spur_gear2d(mod=5, teeth=5, pressure_angle=14.5, shaft_diam=2);
// rack2d(teeth=4, bottom=15, mod=5, pressure_angle=14.5);
// Continues:
// Profile shifting brings with it another complication: in order to maintain the specified clearance, the tips of the
// gear teeth need to be shortened. The shortening factor depends on characteristics of both gears, so it cannot
// be automatically incorporated. (Consider the situation where one gear mates with multiple other gears.) With modest
// profile shifts, you can probably ignore this adjustment, but with more extreme profile shifts, it may be important.
// You can compute the shortening parameter using {{gear_shorten()}}. Note that the actual shortening distance is obtained
// by scaling the shortening factor by the gear's module.
// Figure(2D,Big,NoAxes,VPT=[55.8861,-4.31463,8.09832],VPR=[0,0,0],VPD=325.228): With large profile shifts the teeth need to be shortened or they don't have clearance in the valleys of the teeth in the meshing gear.
// teeth1=25;
// teeth2=19;
// mod=4;
// ps1 = 0.75;
// ps2 = 0.75;
// d = gear_dist(mod=mod, teeth1,teeth2,0,ps1,ps2);
// color("lightblue")
// spur_gear2d(mod=mod,teeth=teeth1,profile_shift=ps1,gear_spin=-90);
// right(d)
// spur_gear2d(mod=mod,teeth=teeth2,profile_shift=ps2,gear_spin=-90);
// right(9)stroke([[1.3*d/2,0],[d/2+4,0]], endcap2="arrow2",color="black");
// fwd(2)right(d/2+25)color("black"){back(4)text("No clearance",size=6);
// fwd(4)text("at tooth tip",size=6);}
// Figure(2D,Big,NoAxes,VPT=[55.8861,-4.31463,8.09832],VPR=[0,0,0],VPD=325.228): Applying the correct shortening factor restores the clearance to its set value.
// teeth1=25;
// teeth2=19;
// mod=4;
// ps1 = 0.75;
// ps2 = 0.75;
// d = gear_dist(mod=mod, teeth1,teeth2,0,ps1,ps2);
// shorten=gear_shorten(teeth1,teeth2,0,ps1,ps2);
// color("lightblue")
// spur_gear2d(mod=mod,teeth=teeth1,profile_shift=ps1,shorten=shorten,gear_spin=-90);
// right(d)
// spur_gear2d(mod=mod,teeth=teeth2,profile_shift=ps2,shorten=shorten,gear_spin=-90);
// right(9)stroke([[1.3*d/2,0],[d/2+4,0]], endcap2="arrow2",color="black");
// fwd(2)right(d/2+25)color("black"){back(4)text("Normal",size=6);
// fwd(4)text("Clearance",size=6);}
// Subsection: Helical Gears
// Helicals gears are a modification of spur gears. They can replace spur gears in any application. The teeth are cut
// following a slanted, helical path. The angled teeth engage more gradually than spur gear teeth, so they run more smoothly
// and quietly. A disadvantage of helical gears is that they have thrust along the axis of the gear that must be
// accomodated. Helical gears also have more sliding friction between the meshing teeth compared to spur gears.
// Figure(3D,Med,NoAxes,VPT=[3.5641,-7.03148,4.86523],VPR=[62.7,0,29.2],VPD=263.285): A Helical Gear
// spur_gear(mod=5,teeth=18,pressure_angle=20,thickness=35,helical=-29,shaft_diam=15,slices=15);
// Continues:
// Helical gears have the same compatibility requirements as spur gears, with the additional requirement that
// the helical angles must be opposite each other, so a gear with a helical angle of 35 must mesh with one
// that has an angle of 35. The industry convention refers to these as left-handed and right handed. In
// this library, positive helical angles produce a left handed gear and negative angles produce a right handed gear.
// Figure(3D,Med,NoAxes,VPT=[73.6023,-29.9518,-12.535],VPR=[76,0,1.2],VPD=610): Left and right handed helical gears at 35 degrees.
// spur_gear(mod=5, teeth=20, helical=35, thickness=70,slices=15);
// right(150)
// spur_gear(mod=5, teeth=20, helical=-35, thickness=70,slices=15);
// down(22)
// left(60)
// fwd(220)
// rot($vpr)
// color("black")text3d("left handed right handed",size=18);
// down(52)
// left(55)
// fwd(220)
// rot($vpr)
// color("black")text3d("helical=35 helical=35",size=18);
// Continues:
// The pitch circle of a helical gear is larger compared to a spur gear
// by the cosine of the helical angle, so you cannot simply drop helical gears in to replace spur gears without
// making other adjustments. This dependence does allow you to make
// make much bigger spacing adjustments than are possible with profile shifting—without changing the tooth count.
// The {{gear_dist()}} function will also compute the appropriate gear spacing for helical gears.
// The effective pressure angle of helical gears is larger than the nominal pressure angle. This can make it possible
// to avoid undercutting without having to use profile shifting, so smaller tooth count gears can be more effective
// using the helical form.
// Figure(Anim,Med,Frames=10,NoAxes,VPT=[43.8006,15.9214,3.52727],VPR=[62.3,0,20.3],VPD=446.129): Meshing compatible helical gears
// zrot($t*360/18)
// spur_gear(mod=5, teeth=18, pressure_angle=20, thickness=25, helical=-29, shaft_diam=15);
// right(gear_dist(mod=5, teeth1=18, teeth2=18, helical=29))
// zrot(360/18/2)
// zrot(-$t*360/18)
// spur_gear(mod=5, teeth=18, pressure_angle=20, thickness=25, helical=29, shaft_diam=15);
// Continues:
// Helical gears can mesh in a second manner that is different from spur gears: they can turn on skew, or crossed axes. These are also
// sometimes called "screw gears". The general requirement for two non-profile-shifted helical gears to mesh is that the angle
// between the gears' axes must equal the sum of the helical angles of the two gears, thus for parallel axes, the helical
// angles must sum to zero. If helical gears are profile shifted, then in addition to adjusting the distance between the
// gears, a small adjustment in the angle is needed, so profile shifted gears won't mesh exactly at the sum of their angles.
// The calculation for gear spacing is different for skew axis gears than for parallel gears, so you do this using {{gear_dist_skew()}},
// and if you use profile shifting, then you can compute the angle using {{gear_skew_angle()}}.
// Figure(Anim,Med,NoAxes,Frames=10,VPT=[44.765,6.09492,-3.01199],VPR=[55.7,0,33.2],VPD=401.289): Two helical gears meshing with axes at a 45 degree angle
// dist = gear_dist_skew(mod=5, teeth1=18, teeth2=18, helical1=22.5,helical2=22.5);
// axiscolor="darkgray";
// down(10)color(axiscolor) cyl(d=15, l=145);
// zrot($t*360/18)
// color("lightblue")spur_gear(mod=5,teeth=18,pressure_angle=20,thickness=25,helical=22.5,shaft_diam=15);
// right(dist)
// xrot(45) {color(axiscolor)cyl(d=15,l=85);
// zrot(360/18/2)
// zrot(-$t*360/18)
// spur_gear(mod=5,teeth=18,pressure_angle=20,thickness=25,helical=22.5,shaft_diam=15);}
// Subsection: Herringbone Gears
// The herringbone gear is made from two stacked helical gears with opposite angles. This design addresses the problem
// of axial forces that afflict helical gears by having one section that slopes to the
// right and another that slopes to the left. Herringbone gears also have the advantage of being self-aligning.
// Figure(3D,Med,NoAxes,VPT=[3.5641,-7.03148,4.86523],VPR=[62.7,0,29.2],VPD=263.285): A herringbone gear
// spur_gear(mod=5, teeth=16, pressure_angle=20, thickness=35, helical=-20, herringbone=true, shaft_diam=15);
// Subsection: Ring Gears (Internal Gears)
// A ring gear (or internal gear) is a gear where the teeth are on the inside of a circle. Such gears must be mated
// to a regular (external) gear, which rotates around the inside.
// Figure(2D,Med,NoAxes,VPT=[0.491171,1.07815,0.495977],VPR=[0,0,0],VPD=292.705): A interior or ring gear (yellow) with a mating spur gear (blue)
// teeth1=18;
// teeth2=30;
// ps1=undef;
// ps2=auto_profile_shift(teeth=teeth1);
// mod=3;
// d = gear_dist(mod=mod, teeth1=teeth1, teeth2=teeth2,profile_shift1=ps1, profile_shift2=ps2,helical=0, internal2=true);
// ang = 0;
// ring_gear2d(mod=mod, teeth=teeth2,profile_shift=ps2,helical=0,backing=4);
// zrot(ang*360/teeth2)
// color("lightblue")
// fwd(d)
// spur_gear2d(mod=mod, teeth=teeth1, profile_shift=ps1,gear_spin=-ang*360/teeth1,helical=0);
// Continues:
// Ring gears are subject to all the usual mesh requirements: the teeth must be the same size, the pressure angles must
// match and they must have opposite helical angles. The {{gear_dist()}} function can give the center separation of
// a ring gear and its mating spur gear. Ring gears have additional complications that tend to arise when the number of
// teeth is small or the teeth counts of the ring gear and spur gear are too close together. The mating spur gear must
// have few enough teeth so that the teeth don't interfere on the other side of the ring. Very small spur gears can interfere
// on the tips of the ring gear's teeth.
// Figure(2D,Med,NoAxes,VPT=[-1.16111,0.0525612,0.495977],VPR=[0,0,0],VPD=213.382): The red regions show interference between the two gears: the 18 tooth spur gear does not fit inside the 20 tooth ring gear.
// teeth1=18;
// teeth2=20;
// ps1=undef;
// ps2=auto_profile_shift(teeth=teeth1);
// mod=3;
// d = gear_dist(mod=mod, teeth1=teeth1, teeth2=teeth2,profile_shift1=ps1, profile_shift2=ps2,helical=0, internal2=true);
// ang = 0;
// color_overlaps(){
// ring_gear2d(mod=mod, teeth=teeth2,profile_shift=ps2,helical=0,backing=4);
// zrot(ang*360/teeth2)
// fwd(d)
// spur_gear2d(mod=mod, teeth=teeth1, profile_shift=ps1,gear_spin=-ang*360/teeth1,helical=0);
// }
// Figure(2D,Big,NoAxes,VPT=[10.8821,-26.1226,-0.0685569],VPD=43.9335,VPR=[0,0,16.8]): Interference at teeth tips, shown in red, with a 5 tooth and 19 tooth gear.
// $fn=128;
// teeth1=5;
// teeth2=19;
// ps1=0;
// ps2=0;
// mod=3;
// d = gear_dist(mod=mod, teeth1=teeth1, teeth2=teeth2,profile_shift1=ps1, profile_shift2=ps2,helical=0, internal2=true);
// ang = 1;
// color_overlaps(){
// ring_gear2d(mod=mod, teeth=teeth2,profile_shift=ps2,helical=0,backing=4);
// zrot(ang*360/teeth2)
// fwd(d)
// spur_gear2d(mod=mod, teeth=teeth1, profile_shift=ps1,gear_spin=-ang*360/teeth1,helical=0);
// }
// Continues:
// The tooth tip interference can often be controlled using profile shifting of the ring gear, but another requirement is
// that the profile shift of the ring gear must be at least as big as the profile shift of the mated spur gear. In order
// to ensure that this condition holds, you may need to use {{auto_profile_shift()}} to find the profile shift that is
// automatically applied to the spur gear you want to use.
// Figure(2D,Med,VPT=[4.02885,-46.6334,1.23363],VPR=[0,0,6.3],VPD=75.2671,NoAxes): Ring gear without profile shifting doesn't have room for the fat profile shifted teeth of the 5-tooth spur gear, with overlaps shown in red.
// $fn=128;
// teeth1=5;
// teeth2=35;
// ps1=undef;
// ps2=0;
// mod=3;
// d=45-.7;
// ang = .5;
// color_overlaps(){
// ring_gear2d(mod=mod, teeth=teeth2,profile_shift=ps2,helical=0,backing=4);
// zrot(ang*360/teeth2)
// fwd(d)
// spur_gear2d(mod=mod, teeth=teeth1, profile_shift=ps1,gear_spin=-ang*360/teeth1,helical=0);
// }
// Figure(2D,Med,VPT=[9.87969,-45.6706,0.60448],VPD=82.6686,VPR=[0,0,11],NoAxes): When the ring gear is profile shifted to match the spur gear, then the gears mesh without interference.
// $fn=128;
// teeth1=5;
// teeth2=35;
// ps1=undef;
// ps2=auto_profile_shift(teeth=teeth1);
// mod=3;
// d = gear_dist(mod=mod, teeth1=teeth1, teeth2=teeth2,profile_shift1=ps1, profile_shift2=ps2,helical=0, internal2=true);
// ang = .5;
// color_overlaps(){
// ring_gear2d(mod=mod, teeth=teeth2,profile_shift=ps2,helical=0,backing=4);
// zrot(ang*360/teeth2)
// fwd(d)
// spur_gear2d(mod=mod, teeth=teeth1, profile_shift=ps1,gear_spin=-ang*360/teeth1,helical=0);
// }
// Figure(3D,Med,NoAxes,VPT=[2.48983,2.10149,0.658081],VPR=[70.4,0,123],VPD=237.091): A helical ring gear (yellow) mating with the compatible spur gear (blue)
// $fn=128;
// teeth1=18;
// teeth2=30;
// ps1=undef;
// ps2=auto_profile_shift(teeth=teeth1);
// mod=3;
// d = gear_dist(mod=mod, teeth1=teeth1, teeth2=teeth2,profile_shift1=ps1, profile_shift2=ps2,helical=30, internal2=true);
// ang = 0;
// ring_gear(mod=mod, teeth=teeth2,profile_shift=ps2,backing=4,helical=30,thickness=15);
// zrot(ang*360/teeth2)
// color("lightblue")
// fwd(d)
// spur_gear(mod=mod, teeth=teeth1, profile_shift=ps1,gear_spin=-ang*360/teeth1,helical=-30,thickness=15);
// Subsection: Worm Drive
// A worm drive is a gear system for connecting skew shafts at 90 degrees. They offer higher load capacity compared to
// crossed helical gears. The assembly is driven by the "worm", which is a gear that resembles a screw.
// Like a screw, it can have one, or several starts. These starts correspond to teeth on a helical gear;
// in fact, the worm can be regarded as a type of helical gear at a very extreme angle, where the teeth wrap
// around the gear. The worm mates with the "worm gear" which is also called the "worm wheel". The worm gear
// resembles a helical gear at a very slight angle.
// Figure(3D,Med,NoAxes,VPT=[38.1941,-7.67869,7.95996],VPR=[56.4,0,25],VPD=361.364): Worm drive assembly, with worm on the left and worm gear (worm wheel) on the right. When the worm turns its screwing action drives the worm gear.
// starts=2;
// ps=0;
// dist_ba=0;
// gear_ba=0;
// worm(
// d=44, // mate_teeth=30,
// circ_pitch=3*PI,
// starts=starts,orient=BACK);
// right(worm_dist(d=44,mod=3,teeth=30, starts=starts,profile_shift=ps,backlash=dist_ba))
// zrot(360/30*.5)
// worm_gear(
// circ_pitch=3*PI,
// teeth=30,
// worm_diam=44,profile_shift=ps,
// worm_starts=starts,backlash=gear_ba);
// Continues:
// A close look at the worm gear reveals that it differs significantly from a helical or spur gear.
// This gear is an "enveloping" gear, which is designed to follow the curved profile of the worm,
// resulting in much better contact between the teeth of the worm and the teeth of the worm gear.
// The worm shown above is a cylindrical worm, which is the most common type.
// It is possible to design the worm to follow the curved shape of its mated gear, resulting
// in an enveloping (also called "globoid") worm. This type of worm makes better contact with
// the worm gear, but is less often used due to manufacturing complexity and consequent expense.
// Figure(3D,Big,NoAxes,VPT=[0,0,0],VPR=[192,0,180],VPD=172.84): A cylindrical worm appears on the left in green. Note it's straight sides. The enveloping (globoid) worm gears appears on the right in green. Note that its sides curve so several teeth can mate with the worm gear, and it requires a complex tooth form
// tilt=20;
// starts=1;
// ps=0;
// pa=27;
// dist_ba=0;
// gear_ba=0;
// xdistribute(spacing=25){
// xflip()yrot(-tilt)
// union(){
// color("lightgreen")
// xrot(90)
// zrot(-90)
// enveloping_worm( mate_teeth=60,$fn=128,
// d=14, pressure_angle=pa, mod=3/2,
// starts=starts);
// right(worm_dist(d=14,mod=3/2,teeth=60, starts=starts,profile_shift=ps,backlash=dist_ba,pressure_angle=pa))
// zrot(360/30*.25)
// worm_gear(
// mod=3/2,pressure_angle=pa,
// teeth=60,crowning=0,
// worm_diam=14,profile_shift=ps,
// worm_starts=starts,backlash=gear_ba);
// }
// yrot(-tilt)
// union(){
// color("lightgreen")
// xrot(90)
// zrot(-90)
// worm(l=43, $fn=128,
// d=14, pressure_angle=pa, left_handed=true,
// mod=3/2,//circ_pitch=3*PI/2,
// starts=starts);
// right(worm_dist(d=14,mod=3/2,teeth=60, starts=starts,profile_shift=ps,backlash=dist_ba,pressure_angle=pa))
// zrot(360/30*.25)
// worm_gear(
// mod=3/2,pressure_angle=pa,
// teeth=60,crowning=0,left_handed=true,
// worm_diam=14,profile_shift=ps,
// worm_starts=starts,backlash=gear_ba);
// }
// }
// Continues:
// As usual, a proper mesh requires that the pressure angles match and the teeth of the worm and worm gear
// are the same size. Additionally the worm gear must be constructed to match the diameter of the worm
// and the number of starts on the worm. Note that the number of starts changes the angle at of the
// teeth on the worm, and hence requires a change to the angle of teeth on the worm gear.
// Of course an enveloping worm needs to know the diameter of the worm gear; you provide this
// information indirectly by giving the number of teeth on the worm gear.
// The {{worm_dist()}} function will give the correct center spacing for the worm from its mating worm gear.
// .
// Worm drives are often "self-locking", which means that torque transmission can occur only from the worm to the worm gear,
// so they must be driven by the worm. Self-locking results from the small lead angle of the worm threads, which produces
// high frictional forces at contact. A multi-start worm has a higher lead angle and as a result is less likely
// to be self-locking, so a multi-start worm can be chosen to avoid self-locking.
// Since self-locking is associated with friction, self-locking drives have lower efficiency,
// usually less than 50%. Worm drive efficiency can exceed 90% if self-locking is not required. One consideration
// with self-locking systems is that if the worm gear moves a large mass and the drive is suddenly shut off, the
// worm wheel is still trying to move due to inertia, which can create large loads that fracture the worm.
// In such cases, the worm cannot be stopped abruptly but must rotate a little further (called "over travel")
// after switching off the drive.
// Subsection: Bevel Gears
// Bevel gearing is another way of dealing with intersecting gear shafts. For bevel gears, the teeth centers lie on
// the surface of an imaginary cone, which is the "pitch cone" of the bevel gear. Two bevel gears can mesh when their pitch cone
// apexes coincide and the cones touch along their length. The teeth of bevel gears shrink as they get closer to the center of the gear.
// Tooth dimensions and pitch diameter (the base of the pitch cone) are referenced to the outer end of the teeth.
// Note that the pitch radius, computed the same was as for other gears, gives the radius of the pitch cone's base.
// Bevel gears can be made with straight teeth, analogous to spur gears, and with the
// same disadvantage of sudden full contact that is noisy. Spiral teeth are analogous to helical
// teeth on cylindrical gears: the teeth engage gradually and smoothly, transmitting motion more smoothly
// and quietly. Also like helical gears, they have the disadvantage of introducing axial forces, and
// usually they can only operate in one rotation direction.
// A third type of tooth is the zerol tooth, which has curved teeth like the spiral teeth,
// but with a zero angle. These share advantages of straight teeth and spiral teeth: they are quiet like
// straight teeth but they lack the axial thrust of spiral gears, and they can operate in both directions.
// They are also reportedly stronger than either spiral or bevel gears.
// Figure(3D,Med,VPT=[-5.10228,-3.09311,3.06426],VPR=[67.6,0,131.9],VPD=237.091,NoAxes): Straight tooth bevel gear with 45 degree angled teeth. To get a gear like this you must specify a spiral angle of zero and a cutter radius of zero. This gear would mate with a copy of itself and would change direction of rotation without changing the rotation rate.
// bevel_gear(mod=3,teeth=35,mate_teeth=35,face_width=20,spiral=0,cutter_radius=0);
// Figure(3D,Med,VPT=[-5.10228,-3.09311,3.06426],VPR=[67.6,0,131.9],VPD=237.091,NoAxes): Straight tooth bevel gear with 45 degree angled teeth. A gear like this has a positive spiral angle, which determines how sloped the teeth are and a positive cutter radius, which determines how curved the teeth are.
// bevel_gear(mod=3,teeth=35,mate_teeth=35,face_width=20,slices=12);
// Figure(3D,Med,VPT=[-5.10228,-3.09311,3.06426],VPR=[67.6,0,131.9],VPD=237.091,NoAxes): Zerol tooth bevel gear with 45 degree angled teeth. A gear like this has a spiral angle of zero, but a positive cutter radius, which determines how curved the teeth are.
// bevel_gear(mod=3,teeth=35,mate_teeth=35,face_width=20,spiral=0,slices=12);
// Continues:
// Bevel gears have demanding requirements for successful mating of two gears. Of course the tooth size
// and pressure angle must match. But beyond that, their pitch cones have to meet at their points.
// This means that if you specify the tooth counts
// of two gears and the desired shaft angle, then that information completely determines the pitch cones, and hence
// the geometry of the gear. You cannot simply mate two arbitary gears that have the same tooth size
// and pressure angle like you can with helical gears: the gears must be designed in pairs to work together.
// .
// It is most common to design bevel gears so operate with their shafts at 90 degree angles, but
// this is not required, and you can design pairs of bevel gears for any desired shaft angle.
// Note, however, that some shaft angles may result in extreme bevel gear configurations.
// Figure(3D,Med,NoAxes,VPT=[-1.42254,-1.98925,13.5702],VPR=[76,0,145],VPD=263.435): Two zerol bevel gears mated with shafts at 90 degrees.
// bevel_gear(mod=3,teeth=35,face_width=undef,spiral=0,mate_teeth=15,backing=3);
// cyl(h=28,d=3,$fn=16,anchor=BOT);
// color("lightblue")left(pitch_radius(mod=3,teeth=35))up(pitch_radius(mod=3,teeth=15))
// yrot(90){zrot(360/15/2)bevel_gear(mod=3,teeth=15,face_width=undef,spiral=0,right_handed=true,mate_teeth=35);
// cyl(h=57,d=3,$fn=16,anchor=BOT);}
// Figure(3D,Med,NoAxes,VPT=[2.01253,-0.673328,8.98056],VPD=263.435,VPR=[79.5,0,68.6]): Two zerol bevel gears mated with shafts at a 115.38 deg angle. This is a planar bevel gear. The axes intersect on the pitch base of the yellow gear. If the blue gear is tipped slightly more its shaft will intersect the shaft of the yellow gear underneath that gear's pitch base, indicating an impossible angle for a normal bevel gear at this pair of teeth counts.
// ang=acos(-15/35);
// bevel_gear(mod=3,35,15,ang,spiral=0,face_width=undef,backing=5,anchor="apex")
// cyl(h=25,d=3,$fn=16,anchor=BOT);
// color("lightblue")
// xrot(ang)
// bevel_gear(mod=3,15,35,ang,spiral=0,face_width=undef,right_handed=true,anchor="apex")
// cyl(h=70,d=3,$fn=16,anchor=BOT);
// Continues:
// In the above figure you can see a flat bevel gear. Such a bevel gear is called a planar bevel gear or
// sometimes also a crown gear. The latter term may be confusing because it also refers to a similar looking
// but very different type of gear that is described below. A planar bevel gear can only mate with another
// compatible bevel gear. It has a degenerate cone with its apex on the gear itself, so the mating pinion gear cannot
// mate at a 90 degree angle because if it did, its cone could not meet the center of the planar bevel gear.
// If you request a larger shaft angle, the teeth of the bevel gear will tilt inward, producing an internal bevel gear.
// Gears with this design are rarely used. The mate of an interior gear is always an exterior gear.
// Figure(Med,VPT=[-1.07698,0.67915,-2.25898],VPD=263.435,VPR=[69.7,0,49.3],NoAxes): Internal bevel gear (yellow) mated to an external bevel gear (blue) to achieve a 135 degree shaft angle.
// ang=135;
// bevel_gear(mod=3,35,15,ang,spiral=0,cone_backing=false);
// down(15)cyl(h=40,d=3,$fn=16,anchor=BOT);
// color("lightblue")
// back(pitch_radius(mod=3,teeth=35)+pitch_radius(mod=3,teeth=15))
// xrot(ang,cp=[0,-pitch_radius(mod=3,teeth=15),0]){
// bevel_gear(mod=3,15,35,ang,right_handed=true,spiral=0);
// cyl(h=40,d=3,$fn=16,anchor=BOT);
// }
// Subsection: Crown Gears (Face Gears)
// Crown gears, sometimes called Face Crown Gears or just Face Gears, are gears with teeth pointing straight up so
// the gear resembles a crown. This type of gear is not the same as a bevel gear with vertical teeth, which would mate
// to another bevel gear. A crown gear mates to a spur gear at a ninety degree angle. A feature of the crown gear assembly
// is that the spur gear can shift along its axis without affecting the mesh.
// Figure(Med,NoAxes,VPT=[-2.19006,-1.67419,-4.49379],VPR=[67.6,0,131.9],VPD=113.4): A Crown or Face gear with its mating spur gear in blue.
// crown_gear(mod=1, teeth=32, backing=3, face_width=7);
// color("lightblue")
// back(pitch_radius(mod=1,teeth=32)+7/2)
// up(gear_dist(mod=1,teeth1=0,teeth2=9))spur_gear(mod=1, teeth=9,orient=BACK,thickness=7,gear_spin=360/9/2);
// Continues:
// When constructing a crown gear you need to make it with the same given pressure and and tooth size as
// the spur gear you wish to mate to it. However, the teeth of a crown gear have pressure angle that varies
// along the width of the tooth. The vertical separation of the spur gear from the crown gear is given
// by {{gear_dist()}} where you treat the crown gear as a rack. The inner radius of the teeth on the
// crown gear is the pitch radius determined by the gear's tooth size and number of teeth. The face width
// of a crown gear is limited by geometry, so if you make it too large you will get an error.
// .
// Note that the geometry of these crown gears is tricky and not well documented by sources we have found.
// If you know something about crown gears that could improve the implementation, please open an issue
// on github.
// Section: Backlash (Fitting Real Gears Together)
// You may have noticed that the example gears shown fit together perfectly, making contact on both sides of
// the teeth. Real gears need space between the teeth to prevent the gears from jamming, to provide space
// for lubricant, and to provide allowance for fabrication error. This space is called backlash. Excessive backlash
// is undesirable, especially if the drive reverses frequently.
// .
// Backlash can be introduced in two ways. One is to make the teeth narrower, so the gaps between the teeth are
// larger than the teeth. Alternatively, you can move the gears farther apart than their ideal spacing.
// Backlash can be measured in several different ways. The gear modules in this library accept a backlash
// parameter which specifies backlash as a circular distance at the pitch circle. The modules narrow
// the teeth by the amount specified, which means the spaces between the teeth grow larger. Of course, if you apply
// backlash to both gears then the total backlash in the system is the combined amount from both gears.
// Usually it is best to apply backlash symmetrically to both gears, but if one gear is very small it may
// be better to place the backlash entirely on the larger gear to avoid weakening the teeth of the small gear.
// Figure(2D,Big,VPT=[4.5244,64.112,0.0383045],VPR=[0,0,0],VPD=48.517,NoAxes): Backlash narrows the teeth by the specified length along the pitch circle. Below the ideal gear appears in the lighter color and the darker color shows the same gear with a very large backlash, which appears with half of the backlash on either side of the tooth.
// teeth1=20;
// mod=5;
// r1 = pitch_radius(mod=mod,teeth=teeth1,helical=40);
// bang=4/(2*PI*r1) * 360 ;
// zrot(-180/teeth1*.5){
// color("white")
// dashed_stroke(arc(r=r1, n=30, angle=[80,110]), width=.05);
// spur_gear2d(mod=mod, teeth=teeth1,backlash=0+.5*0,profile_shift="auto",gear_spin=180/teeth1*.5,helical=40);
// %spur_gear2d(mod=mod, teeth=teeth1,backlash=4+.5*0,profile_shift="auto",gear_spin=180/teeth1*.5,helical=40);
// color("black")stroke(arc(n=32,r=r1,angle=[90+bang/2,90]),width=.1,endcaps="arrow2");
// }
// color("black")back(r1+.25)right(5.5)text("backlash/2",size=1);
// Figure(2D,Med,VPT=[0.532987,50.0891,0.0383045],VPR=[0,0,0],VPD=53.9078): Here two gears appear together with a more reasonable backlash applied to both gears. Again the lighter color shows the ideal gears and the darker shade shows the gear with backlash. Note that in this example, backlash is present on both of the meshing gears, so the total backlash of the system is the combined backlash from both gears.
// teeth1=20;teeth2=33;
// mod=5;
// ha=0;
// r1 = pitch_radius(mod=mod,teeth=teeth1,helical=ha);
// r2=pitch_radius(mod=mod,teeth=teeth2,helical=ha);
// bang=4/(2*PI*r1) * 360 ;
//
// back(r1+pitch_radius(mod=mod,teeth=teeth2,helical=ha)){
// spur_gear2d(mod=mod, teeth=teeth2,backlash=.5*0,helical=ha,gear_spin=-180/teeth2/2);
// %spur_gear2d(mod=mod, teeth=teeth2,backlash=1,helical=ha,gear_spin=-180/teeth2/2);
// }
// {
// spur_gear2d(mod=mod, teeth=teeth1,backlash=0+.5*0,profile_shift=0,gear_spin=180/teeth1*.5,helical=ha);
// %spur_gear2d(mod=mod, teeth=teeth1,backlash=1+.5*0,profile_shift=0,gear_spin=180/teeth1*.5,helical=ha);
// *color("white"){
// dashed_stroke(arc(r=r1, n=30, angle=[80,110]), width=.05);
// back(r1+r2)
// dashed_stroke(arc(r=r2, n=30, angle=[-80,-110]), width=.05);
// }
// //color("black")stroke(arc(n=32,r=r1,angle=[90+bang/2,90]),width=.1,endcaps="arrow2");
// }
// Figure(2D,Med,VPT=[0.532987,50.0891,0.0383045],VPR=[0,0,0],VPD=53.9078): Here the same gears as in the previous figure appear with backlash applied using the `backlash` parameter to {{gear_dist()}} to shift them apart. The original ideal gears are in the lighter shade and the darker colored gears have been separated to create the backlash.
// teeth1=20;teeth2=33;
// mod=5;
// ha=0;
// r1 = pitch_radius(mod=mod,teeth=teeth1,helical=ha);
// r2 = pitch_radius(mod=mod,teeth=teeth2,helical=ha);
// bang=4/(2*PI*r1) * 360 ;
// shift = 1 * cos(ha)/2/tan(20);
// back(r1+pitch_radius(mod=mod,teeth=teeth2,helical=ha)){
// zrot(-180/teeth2/2){
// %back(shift)spur_gear2d(mod=mod, teeth=teeth2,backlash=0,helical=ha);
// spur_gear2d(mod=mod, teeth=teeth2,backlash=0,helical=ha);
// }
// }
// zrot(180/teeth1*.5){
// %fwd(shift)spur_gear2d(mod=mod, teeth=teeth1,backlash=0+.5*0,profile_shift=0,helical=ha);
// spur_gear2d(mod=mod, teeth=teeth1,backlash=0,profile_shift=0,helical=ha);
// }
// Section: Gears
// Function&Module: spur_gear()
// Synopsis: Creates a spur gear, helical gear, or internal ring gear.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: rack(), spur_gear(), spur_gear2d(), bevel_gear()
// Usage: As a Module
// spur_gear(circ_pitch, teeth, [thickness], [helical=], [pressure_angle=], [profile_shift=], [backlash=], [shaft_diam=], [hide=], [clearance=], [slices=], [internal=], [herringbone=]) [ATTACHMENTS];
// spur_gear(mod=|diam_pitch=, teeth=, [thickness=], ...) [ATTACHMENTS];
// Usage: As a Function
// vnf = spur_gear(circ_pitch, teeth, [thickness], ...);
// vnf = spur_gear(mod=|diam_pitch=, teeth=, [thickness=], ...);
// Description:
// Creates a involute spur gear, helical gear, herringbone gear, or a mask for an internal ring gear.
// For more information about gears, see [A Quick Introduction to Gears](gears.scad#section-a-quick-introduction-to-gears).
// You must specify the teeth size using either `mod=`, `circ_pitch=` or `diam_pitch=`, and you
// must give the number of teeth of the gear. Spur gears have straight teeth and
// mesh together on parallel shafts without creating any axial thrust. The teeth engage suddenly across their
// entire width, creating stress and noise. Helical gears have angled teeth and engage more gradually, so they
// run more smoothly and quietly, however they do produce thrust along the gear axis. This can be
// circumvented using herringbone or double helical gears, which have no axial thrust and also self-align.
// Helical gears can mesh along shafts that are not parallel, where the angle between the shafts is
// the sum of the helical angles of the two gears.
// .
// The module creates the gear in the XY plane, centered on the origin, with one tooth centered on the positive Y axis.
// In order for two gears to mesh they must have the same tooth size and `pressure_angle`, and
// generally the helical angles should be of opposite sign.
// The usual pressure angle (and default) is 20 degrees. Another common value is 14.5 degrees.
// Ideally the teeth count of two meshing gears will be relatively prime because this ensures that
// every tooth on one gear will meet every tooth on the other, creating even wear.
// .
// The "pitch circle" of the gear is a reference circle where the circular pitch is defined that
// is used to construct the gear. It runs approximately through the centers of the teeth.
// Two basic gears will mesh when their pitch circles are tangent. Anchoring for these gears is
// done on the pitch circle by default, so basic gears can be meshed using anchoring.
// However, when a gear has a small number of teeth, the basic gear form will result in undercutting,
// which weakens the teeth. To avoid this, profile shifting is automatically applied and in this
// case, the distance between the gears is a complicated calculation and must be determined using {{gear_dist()}}.
// If you wish to override this correction, you can use `profile_shift=0`, or set it to a specific
// value like 0.5. Another complication with profile shifted gears is that the tips may be too long,
// which can eat into the clearance space. To address this problem you can use the `shorten` parameter,
// which you can compute using {{gear_shorten()}}.
// .
// Helical gears can mesh with skew or crossed axes, a configuration sometimes called "screw gears".
// Without profile shifting, that angle is the sum of the helical angles.
// With profile shifting it is slightly different and is given by {{gear_skew_angle()}}.
// These gears still mesh on the pitch circle when they are not profile shifted, but the correction to
// gear separation for a proper mesh of profile shifted gears is different for skew gears and is
// computed using {{gear_dist_skew()}}.
// .
// To create space for gears to mesh in practice you will need to set a positive value for backlash, or
// use the `backlash` argument to {{gear_dist()}}.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// teeth = Total number of teeth around the entire perimeter
// thickness = Thickness of gear. Default: 10
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// helical = Teeth spiral around the gear at this angle, positive for left handed, negative for right handed. Default: 0
// herringbone = If true, and helical is set, creates a herringbone gear. Default: False
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// profile_shift = Profile shift factor x. Default: "auto"
// shorten = Shorten gear tips by the module times this value. Needed for large profile shifted gears. Default: 0
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// shaft_diam = Diameter of the hole in the center. Default: 0 (no shaft hole)
// hide = Number of teeth to delete to make this only a fraction of a circle. Default: 0
// gear_spin = Rotate gear and children around the gear center, regardless of how gear is anchored. Default: 0
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: mod/4
// slices = Number of vertical layers to divide gear into. Useful for refining gears with `helical`.
// internal = If true, create a mask for difference()ing from something else.
// atype = Set to "root", "tip" or "pitch" to determine anchoring circle. Default: "pitch"
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Side Effects:
// If internal is true then the default tag is "remove"
// Anchor Types:
// root = anchor on the root circle
// pitch = anchor on the pitch circle (default)
// tip = anchor on the tip circle
// Example: Spur Gear
// spur_gear(circ_pitch=5, teeth=20, thickness=8, shaft_diam=5);
// Example: Metric Gear
// spur_gear(mod=2, teeth=20, thickness=8, shaft_diam=5);
// Example: Helical Gear
// spur_gear(
// circ_pitch=5, teeth=20, thickness=10,
// shaft_diam=5, helical=-30, slices=12,
// $fa=1, $fs=1
// );
// Example: Herringbone Gear
// spur_gear(
// circ_pitch=5, teeth=20, thickness=10, shaft_diam=5,
// helical=30, herringbone=true, slices=5
// );
// Example(Med,VPT=[-0.0213774,2.42972,-0.2709],VPR=[36.1,0,20.1],VPD=74.3596): Effects of Profile Shifting.
// circ_pitch=5; teeth=7; thick=10; shaft=5; strokewidth=0.2;
// pr = pitch_radius(circ_pitch, teeth);
// left(10) {
// profile_shift = 0;
// d = gear_dist(circ_pitch=circ_pitch,teeth,0,profile_shift1=profile_shift);
// back(d) spur_gear(circ_pitch, teeth, thick, shaft, profile_shift=profile_shift);
// rack(circ_pitch, teeth=3, thickness=thick, orient=BACK);
// color("black") up(thick/2) linear_extrude(height=0.1) {
// back(d) dashed_stroke(circle(r=pr), width=strokewidth, closed=true);
// dashed_stroke([[-7.5,0],[7.5,0]], width=strokewidth);
// }
// }
// right(10) {
// profile_shift = 0.59;
// d = gear_dist(circ_pitch=circ_pitch,teeth,0,profile_shift1=profile_shift);
// back(d) spur_gear(circ_pitch, teeth, thick, shaft, profile_shift=profile_shift);
// rack(circ_pitch, teeth=3, thickness=thick, orient=BACK);
// color("black") up(thick/2) linear_extrude(height=0.1) {
// back(d)
// dashed_stroke(circle(r=pr), width=strokewidth, closed=true);
// dashed_stroke([[-7.5,0],[7.5,0]], width=strokewidth);
// }
// }
// Example(Anim,Med,NoAxes,Frames=8,VPT=[0,30,0],VPR=[0,0,0],VPD=300): Assembly of Gears
// $fn=12;
// n1 = 11; //red gear number of teeth
// n2 = 20; //green gear
// n3 = 6; //blue gear
// n4 = 16; //orange gear
// n5 = 9; //gray rack
// circ_pitch = 9; //all meshing gears need the same `circ_pitch` (and the same `pressure_angle`)
// thickness = 6;
// hole = 3;
// rack_base = 12;
// d12 = gear_dist(circ_pitch=circ_pitch,teeth1=n1,teeth2=n2);
// d13 = gear_dist(circ_pitch=circ_pitch,teeth1=n1,teeth2=n3);
// d14 = gear_dist(circ_pitch=circ_pitch,teeth1=n1,teeth2=n4);
// d1r = gear_dist(circ_pitch=circ_pitch,teeth1=n1,teeth2=0);
// a1 = $t * 360 / n1;
// a2 = -$t * 360 / n2 + 180/n2;
// a3 = -$t * 360 / n3 - 3*90/n3;
// a4 = -$t * 360 / n4 - 3.5*180/n4;
// color("#f77") zrot(a1) spur_gear(circ_pitch,n1,thickness,hole);
// color("#7f7") back(d12) zrot(a2) spur_gear(circ_pitch,n2,thickness,hole);
// color("#77f") right(d13) zrot(a3) spur_gear(circ_pitch,n3,thickness,hole);
// color("#fc7") left(d14) zrot(a4) spur_gear(circ_pitch,n4,thickness,hole,hide=n4-3);
// color("#ccc") fwd(d1r) right(circ_pitch*$t)
// rack(pitch=circ_pitch,teeth=n5,thickness=thickness,width=rack_base,anchor=CENTER,orient=BACK);
// Example(NoAxes,VPT=[1.13489,-4.48517,1.04995],VPR=[55,0,25],VPD=139.921): Helical gears meshing with non-parallel shafts
// ang1 = 30;
// ang2 = 10;
// circ_pitch = 5;
// n = 20;
// dist = gear_dist_skew(
// circ_pitch=circ_pitch,
// teeth1=n, teeth2=n,
// helical1=ang1, helical2=ang2);
// left(dist/2) spur_gear(
// circ_pitch, teeth=n, thickness=10,
// shaft_diam=5, helical=ang1, slices=12,
// gear_spin=-90
// );
// right(dist/2)
// xrot(ang1+ang2)
// spur_gear(
// circ_pitch=circ_pitch, teeth=n, thickness=10,
// shaft_diam=5, helical=ang2, slices=12,
// gear_spin=90-180/n
// );
// Example(Anim,Big,NoAxes,Frames=36,VPT=[0,0,0],VPR=[55,0,25],VPD=220): Planetary Gear Assembly
// $fn=128;
// rteeth=56; pteeth=16; cteeth=24;
// circ_pitch=5; thick=10; pa=20;
// gd = gear_dist(circ_pitch=circ_pitch, cteeth, pteeth);
// ring_gear(
// circ_pitch=circ_pitch,
// teeth=rteeth,
// thickness=thick,
// pressure_angle=pa);
// for (a=[0:3]) {
// zrot($t*90+a*90) back(gd) {
// color("green")
// spur_gear(
// circ_pitch=circ_pitch,
// teeth=pteeth,
// thickness=thick,
// shaft_diam=5,
// pressure_angle=pa,
// spin=-$t*90*rteeth/pteeth);
// }
// }
// color("orange")
// zrot($t*90*rteeth/cteeth+$t*90+180/cteeth)
// spur_gear(
// circ_pitch=circ_pitch,
// teeth=cteeth,
// thickness=thick,
// shaft_diam=5,
// pressure_angle=pa);
function spur_gear(
circ_pitch,
teeth,
thickness,
shaft_diam = 0,
hide = 0,
pressure_angle,
clearance,
backlash = 0.0,
helical,
interior,
internal,
profile_shift="auto",
slices,
herringbone=false,
shorten=0,
diam_pitch,
mod,
pitch,
gear_spin = 0,
atype = "pitch",
anchor = CENTER,
spin = 0,
orient = UP
) =
let(
dummy = !is_undef(interior) ? echo("In spur_gear(), the argument 'interior=' has been deprecated, and may be removed in the future. Please use 'internal=' instead."):0,
internal = first_defined([internal,interior,false]),
circ_pitch = _inherit_gear_pitch("spur_gear()", pitch, circ_pitch, diam_pitch, mod),
PA = _inherit_gear_pa(pressure_angle),
helical = _inherit_gear_helical(helical, invert=!internal),
thickness = _inherit_gear_thickness(thickness)
)
assert(is_integer(teeth) && teeth>3)
assert(is_finite(thickness) && thickness>0)
assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(is_integer(hide) && hide>=0 && hide<teeth)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_bool(herringbone))
assert(slices==undef || (is_integer(slices) && slices>0))
assert(is_finite(gear_spin))
let(
profile_shift = auto_profile_shift(teeth,PA,helical,profile_shift=profile_shift),
pr = pitch_radius(circ_pitch, teeth, helical),
or = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=internal,shorten=shorten),
rr = _root_radius(circ_pitch, teeth, clearance, profile_shift=profile_shift, internal=internal),
anchor_rad = atype=="pitch" ? pr
: atype=="tip" ? or
: atype=="root" ? rr
: assert(false,"atype must be one of \"root\", \"tip\" or \"pitch\""),
circum = 2 * PI * pr,
twist = 360*thickness*tan(helical)/circum,
slices = default(slices, ceil(twist/360*segs(pr)+1)),
rgn = spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
hide = hide,
helical = helical,
clearance = clearance,
backlash = backlash,
internal = internal,
shorten = shorten,
profile_shift = profile_shift,
shaft_diam = shaft_diam
),
rvnf = herringbone
? zrot(twist/2, p=linear_sweep(rgn, height=thickness, twist=twist, slices=slices, center=true))
: let(
wall_vnf = linear_sweep(rgn, height=thickness/2, twist=twist/2, slices=ceil(slices/2), center=false, caps=false),
cap_vnf = vnf_from_region(rgn, transform=up(thickness/2)*zrot(twist/2))
)
vnf_join([
wall_vnf, zflip(p=wall_vnf),
cap_vnf, zflip(p=cap_vnf),
]),
vnf = zrot(gear_spin, p=rvnf)
) reorient(anchor,spin,orient, h=thickness, r=anchor_rad, p=vnf);
module spur_gear(
circ_pitch,
teeth,
thickness,
shaft_diam = 0,
hide = 0,
pressure_angle,
clearance,
backlash = 0.0,
helical,
internal,
interior,
profile_shift="auto",
slices,
herringbone=false,
shorten=0,
pitch,
diam_pitch,
mod,
atype="pitch",
gear_spin = 0,
anchor = CENTER,
spin = 0,
orient = UP
) {
dummy = !is_undef(interior) ? echo("In spur_gear(), the argument 'interior=' has been deprecated, and may be removed in the future. Please use 'internal=' instead."):0;
internal = first_defined([internal,interior,false]);
circ_pitch = _inherit_gear_pitch("spur_gear()", pitch, circ_pitch, diam_pitch, mod);
PA = _inherit_gear_pa(pressure_angle);
helical = _inherit_gear_helical(helical, invert=!internal);
thickness = _inherit_gear_thickness(thickness);
checks =
assert(is_integer(teeth) && teeth>3)
assert(is_finite(thickness) && thickness>0)
assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(is_integer(hide) && hide>=0 && hide<teeth)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_bool(herringbone))
assert(slices==undef || (is_integer(slices) && slices>0))
assert(is_finite(gear_spin));
profile_shift = auto_profile_shift(teeth,PA,helical,profile_shift=profile_shift);
pr = pitch_radius(circ_pitch, teeth, helical);
or = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=internal,shorten=shorten);
rr = _root_radius(circ_pitch, teeth, clearance, profile_shift=profile_shift, internal=internal);
anchor_rad = atype=="pitch" ? pr
: atype=="tip" ? or
: atype=="root" ? rr
: assert(false,"atype must be one of \"root\", \"tip\" or \"pitch\"");
circum = 2 * PI * pr;
twist = 360*thickness*tan(helical)/circum;
slices = default(slices, ceil(twist/360*segs(pr)+1));
default_tag("remove", internal) {
attachable(anchor,spin,orient, r=anchor_rad, l=thickness) {
zrot(gear_spin)
if (herringbone) {
zflip_copy() down(0.01)
linear_extrude(
height=thickness/2+0.01, center=false,
twist=twist/2, slices=ceil(slices/2),
convexity=teeth/2
) {
spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
hide = hide,
helical = helical,
clearance = clearance,
backlash = backlash,
internal = internal,
shorten = shorten,
profile_shift = profile_shift,
shaft_diam = shaft_diam
);
}
} else {
zrot(twist/2)
linear_extrude(
height=thickness, center=true,
twist=twist, slices=slices,
convexity=teeth/2
) {
spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
hide = hide,
helical = helical,
clearance = clearance,
backlash = backlash,
internal = internal,
profile_shift = profile_shift,
shaft_diam = shaft_diam
);
}
}
union() {
$parent_gear_type = "spur";
$parent_gear_pitch = circ_pitch;
$parent_gear_teeth = teeth;
$parent_gear_pa = PA;
$parent_gear_helical = helical;
$parent_gear_thickness = thickness;
union() children();
}
}
}
}
// Function&Module: spur_gear2d()
// Synopsis: Creates a 2D spur gear or internal ring gear.
// SynTags: Geom, Region
// Topics: Gears, Parts
// See Also: rack(), spur_gear(), spur_gear2d(), bevel_gear()
// Usage: As Module
// spur_gear2d(circ_pitch, teeth, [pressure_angle=], [profile_shift=], [shorten=], [hide=], [shaft_diam=], [clearance=], [backlash=], [internal=]) [ATTACHMENTS];
// spur_gear2d(mod=|diam_pitch=, teeth=, [pressure_angle=], [profile_shift=], [shorten=], [hide=], [shaft_diam=], [clearance=], [backlash=], [internal=]) [ATTACHMENTS];
// Usage: As Function
// rgn = spur_gear2d(circ_pitch, teeth, [pressure_angle=], [profile_shift=], [shorten=], [hide=], [shaft_diam=], [clearance=], [backlash=], [internal=]);
// rgn = spur_gear2d(mod=, teeth=, [pressure_angle=], [profile_shift=], [shorten=], [hide=], [shaft_diam=], [clearance=], [backlash=], [internal=]);
// Description:
// Creates a 2D involute spur gear, or a mask for an internal ring gear.
// For more information about gears, see [A Quick Introduction to Gears](gears.scad#section-a-quick-introduction-to-gears).
// You must specify the teeth size using either `mod=`, `circ_pitch=` or `diam_pitch=`, and you
// must give the number of teeth.
// .
// The module creates the gear in centered on the origin, with one tooth centered on the positive Y axis.
// In order for two gears to mesh they must have the same tooth size and `pressure_angle`
// The usual pressure angle (and default) is 20 degrees. Another common value is 14.5 degrees.
// Ideally the teeth count of two meshing gears will be relatively prime because this ensures that
// every tooth on one gear will meet every tooth on the other, creating even wear.
// .
// The "pitch circle" of the gear is a reference circle where the circular pitch is defined that
// is used to construct the gear. It runs approximately through the centers of the teeth.
// Two basic gears will mesh when their pitch circles are tangent. Anchoring for these gears is
// done on the pitch circle by default, so basic gears can be meshed using anchoring.
// However, when a gear has a small number of teeth, the basic gear form will result in undercutting,
// which weakens the teeth. To avoid this, profile shifting is automatically applied and in this
// case, the distance between the gears is a complicated calculation and must be determined using {{gear_dist()}}.
// If you wish to override this correction, you can use `profile_shift=0`, or set it to a specific
// value like 0.5. Another complication with profile shifted gears is that the tips may be too long,
// which can eat into the clearance space. To address this problem you can use the `shorten` parameter,
// which you can compute using {{gear_shorten()}}.
// .
// To create space for gears to mesh in practice you will need to set a positive value for backlash, or
// use the `backlash` argument to {{gear_dist()}}.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// teeth = Total number of teeth around the spur gear.
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees.
// profile_shift = Profile shift factor x. Default: "auto"
// shorten = Shorten gear tips by the module times this value. Needed for large profile shifted gears. Default: 0
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// helical = Adjust teeth form (stretch out the teeth) to give the cross section of a gear with this helical angle. Default: 0
// hide = Number of teeth to delete to make this only a fraction of a circle
// gear_spin = Rotate gear and children around the gear center, regardless of how gear is anchored. Default: 0
// clearance = Gap between top of a tooth on one gear and bottom of valley on a meshing gear. Default: mod/4
// internal = If true, create a mask for difference()ing from something else.
// shaft_diam = If given, the diameter of the central shaft hole.
// atype = Set to "root", "tip" or "pitch" to determine anchoring circle. Default: "pitch"
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// Side Effects:
// If internal is true then the default tag is "remove"
// Anchor Types:
// root = anchor on the root circle
// pitch = anchor on the pitch circle (default)
// tip = anchor on the tip circle
// Example(2D): Typical Gear Shape
// spur_gear2d(circ_pitch=5, teeth=20, shaft_diam=5);
// Example(2D): By Metric Module
// spur_gear2d(mod=2, teeth=20, shaft_diam=5);
// Example(2D): By Imperial Gear Pitch
// spur_gear2d(diam_pitch=10, teeth=20, shaft_diam=5);
// Example(2D): Lower Pressure Angle
// spur_gear2d(circ_pitch=5, teeth=20, pressure_angle=14);
// Example(2D): Partial Gear
// spur_gear2d(circ_pitch=5, teeth=20, hide=15, pressure_angle=20);
// Example(2D,Med,VPT=[0.151988,3.93719,1.04995],VPR=[0,0,0],VPD=74.3596): Effects of Profile Shifting.
// circ_pitch=5; teeth=7; shaft=5; strokewidth=0.2;
// module the_gear(profile_shift=0) {
// $fn=72;
// pr = pitch_radius(circ_pitch,teeth);
// mr = gear_dist(circ_pitch=circ_pitch,teeth,profile_shift1=profile_shift,teeth2=0);
// back(mr) {
// spur_gear2d(circ_pitch, teeth, shaft_diam=shaft, profile_shift=profile_shift);
// up(0.1) color("black")
// dashed_stroke(circle(r=pr), width=strokewidth, closed=true);
// }
// }
// module the_rack() {
// $fn=72;
// rack2d(circ_pitch, teeth=3);
// up(0.1) color("black")
// dashed_stroke([[-7.5,0],[7.5,0]], width=strokewidth);
// }
// left(10) { the_gear(0); the_rack(); }
// right(10) { the_gear(0.59); the_rack(); }
// Example(2D): Planetary Gear Assembly
// rteeth=56; pteeth=16; cteeth=24;
// circ_pitch=5; pa=20;
// gd = gear_dist(circ_pitch=circ_pitch, cteeth,pteeth);
// ring_gear2d(
// circ_pitch=circ_pitch,
// teeth=rteeth,
// pressure_angle=pa);
// for (a=[0:3]) {
// zrot(a*90) back(gd) {
// color("green")
// spur_gear2d(
// circ_pitch=circ_pitch,
// teeth=pteeth,
// pressure_angle=pa);
// }
// }
// color("orange")
// zrot(180/cteeth)
// spur_gear2d(
// circ_pitch=circ_pitch,
// teeth=cteeth,
// pressure_angle=pa);
// Example(2D): Called as a Function
// rgn = spur_gear2d(circ_pitch=8, teeth=16, shaft_diam=5);
// region(rgn);
function spur_gear2d(
circ_pitch,
teeth,
hide = 0,
pressure_angle,
clearance,
backlash = 0.0,
internal,
interior,
profile_shift="auto",
helical,
shaft_diam = 0,
shorten = 0,
pitch,
diam_pitch,
mod,
gear_spin = 0,
atype="pitch",
anchor = CENTER,
spin = 0
) = let(
dummy = !is_undef(interior) ? echo("In spur_gear2d(), the argument 'interior=' has been deprecated, and may be removed in the future. Please use 'internal=' instead."):0,
internal = first_defined([internal,interior,false]),
circ_pitch = _inherit_gear_pitch("spur_gear2d()", pitch, circ_pitch, diam_pitch, mod),
PA = _inherit_gear_pa(pressure_angle),
helical = _inherit_gear_helical(helical, invert=!internal)
)
assert(is_integer(teeth) && teeth>3)
assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(is_integer(hide) && hide>=0 && hide<teeth)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_finite(gear_spin))
let(
profile_shift = auto_profile_shift(teeth,PA,helical,profile_shift=profile_shift),
pr = pitch_radius(circ_pitch, teeth, helical=helical),
or = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=internal,shorten=shorten),
rr = _root_radius(circ_pitch, teeth, clearance, profile_shift=profile_shift, internal=internal),
anchor_rad = atype=="pitch" ? pr
: atype=="tip" ? or
: atype=="root" ? rr
: assert(false,"atype must be one of \"root\", \"tip\" or \"pitch\""),
tooth = _gear_tooth_profile(
circ_pitch=circ_pitch,
teeth=teeth,
pressure_angle=PA,
clearance=clearance,
backlash=backlash,
profile_shift=profile_shift,
helical=helical,
shorten=shorten,
internal=internal
),
perim = [
for (i = [0:1:teeth-1-hide])
each zrot(-i*360/teeth+gear_spin, p=tooth),
if (hide>0) [0,0],
],
rgn = [
list_unwrap(deduplicate(perim)),
if (shaft_diam>0 && !hide)
reverse(circle(d=shaft_diam, $fn=max(16,segs(shaft_diam/2)))),
]
) reorient(anchor,spin, two_d=true, r=anchor_rad, p=rgn);
module spur_gear2d(
circ_pitch,
teeth,
hide = 0,
pressure_angle,
clearance,
backlash = 0.0,
internal,
interior,
profile_shift="auto",
helical,
shorten = 0,
shaft_diam = 0,
pitch,
diam_pitch,
mod,
gear_spin = 0,
atype="pitch",
anchor = CENTER,
spin = 0
) {
dummy = !is_undef(interior) ? echo("In spur_gear2d(), the argument 'interior=' has been deprecated, and may be removed in the future. Please use 'internal=' instead."):0;
internal = first_defined([internal,interior,false]);
circ_pitch = _inherit_gear_pitch("spur_gear2d()", pitch, circ_pitch, diam_pitch, mod);
PA = _inherit_gear_pa(pressure_angle);
helical = _inherit_gear_helical(helical, invert=!internal);
checks =
assert(is_integer(teeth) && teeth>3)
assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(is_integer(hide) && hide>=0 && hide<teeth)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_finite(gear_spin));
profile_shift = auto_profile_shift(teeth,PA,helical,profile_shift=profile_shift);
rgn = spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
hide = hide,
pressure_angle = PA,
clearance = clearance,
helical = helical,
backlash = backlash,
profile_shift = profile_shift,
internal = internal,
shorten = shorten,
shaft_diam = shaft_diam
);
pr = pitch_radius(circ_pitch, teeth, helical=helical);
or = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=internal,shorten=shorten);
rr = _root_radius(circ_pitch, teeth, clearance, profile_shift=profile_shift, internal=internal);
anchor_rad = atype=="pitch" ? pr
: atype=="tip" ? or
: atype=="root" ? rr
: assert(false,"atype must be one of \"root\", \"tip\" or \"pitch\"");
attachable(anchor,spin, two_d=true, r=anchor_rad) {
zrot(gear_spin) region(rgn);
union() {
$parent_gear_type = "spur2D";
$parent_gear_pitch = circ_pitch;
$parent_gear_teeth = teeth;
$parent_gear_pa = PA;
$parent_gear_helical = helical;
$parent_gear_thickness = 0;
union() children();
}
}
}
// Module: ring_gear()
// Synopsis: Creates a 3D ring gear.
// SynTags: Geom
// Topics: Gears, Parts
// See Also: rack(), ring_gear2d(), spur_gear(), spur_gear2d(), bevel_gear()
// Usage:
// ring_gear(circ_pitch, teeth, thickness, [backing|od=|or=|width=], [pressure_angle=], [helical=], [herringbone=], [profile_shift=], [clearance=], [backlash=]) [ATTACHMENTS];
// ring_gear(mod=, teeth=, thickness=, [backing=|od=|or=|width=], [pressure_angle=], [helical=], [herringbone=], [profile_shift=], [clearance=], [backlash=]) [ATTACHMENTS];
// ring_gear(diam_pitch=, teeth=, thickness=, [backing=|od=|or=|width=], [pressure_angle=], [helical=], [herringbone=], [profile_shift=], [clearance=], [backlash=]) [ATTACHMENTS];
// Description:
// Creates a 3D involute ring gear.
// Meshing gears must have the same tooth size, pressure angle and helical angle as usual.
// Additionally, you must have more teeth on an internal gear than its mating external gear, and
// the profile shift on the ring gear must be at least as big as the profile shift on the mating gear.
// You may need to use {{auto_profile_shift()}} to find this value if your mating gear has a small number of teeth.
// The gear spacing is given by {{gear_dist()}}.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// teeth = Total number of teeth around the spur gear.
// thickness = Thickness of ring gear
// backing = The width of the ring gear backing. Default: height of teeth
// ---
// od = outer diameter of the ring
// or = outer radius of the ring
// width = width of the ring, measuring from tips of teeth to outside of ring.
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees.
// helical = The angle of the rack teeth away from perpendicular to the gear axis of rotation. Stretches out the tooth shapes. Used to match helical spur gear pinions. Default: 0
// herringbone = If true, and helical is set, creates a herringbone gear.
// profile_shift = Profile shift factor x for tooth profile. Default: 0
// clearance = Gap between top of a tooth on one gear and bottom of valley on a meshing gear (in millimeters)
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Example:
// ring_gear(circ_pitch=5, teeth=48, thickness=10);
// Example: Adjusting Backing
// ring_gear(circ_pitch=5, teeth=48, thickness=10, backing=30);
// Example(Med): Adjusting Pressure Angle
// ring_gear(circ_pitch=5, teeth=48, thickness=10, pressure_angle=28);
// Example(Med): Tooth Profile Shifting
// ring_gear(circ_pitch=5, teeth=48, thickness=10, profile_shift=0.5);
// Example(Med): Helical Ring Gear
// ring_gear(circ_pitch=5, teeth=48, thickness=15, helical=30);
// Example(Med): Herringbone Ring Gear
// ring_gear(circ_pitch=5, teeth=48, thickness=30, helical=30, herringbone=true);
module ring_gear(
circ_pitch,
teeth,
thickness = 10,
backing,
pressure_angle,
helical,
herringbone = false,
profile_shift=0,
clearance,
backlash = 0.0,
or,od,width,
pitch,
diam_pitch,
mod,
slices,
gear_spin = 0,
anchor = CENTER,
atype = "pitch",
spin = 0,
orient = UP
) {
circ_pitch = _inherit_gear_pitch("ring_gear()",pitch, circ_pitch, diam_pitch, mod);
PA = _inherit_gear_pa(pressure_angle);
helical = _inherit_gear_helical(helical); //Maybe broken???
thickness = _inherit_gear_thickness(thickness);
checks =
assert(in_list(atype,["outside","pitch"]))
assert(is_finite(profile_shift), "Profile shift for ring gears must be numerical")
assert(is_integer(teeth) && teeth>3)
assert(is_finite(thickness) && thickness>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(is_finite(helical) && abs(helical)<90)
assert(is_bool(herringbone))
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(slices==undef || (is_integer(slices) && slices>0))
assert(num_defined([backing,or,od,width])<=1, "Cannot define more than one of backing, or, od and width")
assert(is_finite(gear_spin));
pr = pitch_radius(circ_pitch, teeth, helical=helical);
ar = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=true);
rr=_root_radius(circ_pitch, teeth, clearance, profile_shift=profile_shift, internal=true);
or = is_def(or) ?
assert(is_finite(or) && or>ar, "or is invalid or too small for teeth")
or
: is_def(od) ?
assert(is_finite(od) && od>2*ar, "od is invalid or too small for teeth")
od/2
: is_def(width) ?
assert(is_finite(width) && width>ar-rr, "width is invalid or too small for teeth")
rr+width
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
ar+backing
: 2*ar - rr; // default case
circum = 2 * PI * pr;
twist = 360*thickness*tan(-helical)/circum;
slices = default(slices, ceil(twist/360*segs(pr)+1));
attachable(anchor,spin,orient, h=thickness, r=atype=="outside"?or:pr) {
zrot(gear_spin)
if (herringbone) {
zflip_copy() down(0.01)
linear_extrude(height=thickness/2, center=false, twist=twist/2, slices=ceil(slices/2), convexity=teeth/4) {
difference() {
circle(r=or);
spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
helical = helical,
clearance = clearance,
backlash = backlash,
profile_shift = profile_shift,
internal = true
);
}
}
} else {
zrot(twist/2)
linear_extrude(height=thickness,center=true, twist=twist, convexity=teeth/4) {
difference() {
circle(r=or);
spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
helical = helical,
clearance = clearance,
backlash = backlash,
profile_shift = profile_shift,
internal = true
);
}
}
}
children();
}
}
// Module: ring_gear2d()
// Synopsis: Creates a 2D ring gear.
// SynTags: Geom
// Topics: Gears, Parts
// See Also: rack(), spur_gear(), spur_gear2d(), bevel_gear()
// Usage:
// ring_gear2d(circ_pitch, teeth, [backing|od=|or=|width=], [pressure_angle=], [helical=], [profile_shift=], [clearance=], [backlash=]) [ATTACHMENTS];
// ring_gear2d(mod=, teeth=, [backing=|od=|or=|width=], [pressure_angle=], [helical=], [profile_shift=], [clearance=], [backlash=]) [ATTACHMENTS];
// ring_gear2d(diam_pitch=, teeth=, [backing=|od=|or=|width=], [pressure_angle=], [helical=], [profile_shift=], [clearance=], [backlash=]) [ATTACHMENTS];
// Description:
// Creates a 2D involute ring gear.
// Meshing gears must have the same tooth size, pressure angle and helical angle as usual.
// Additionally, you must have more teeth on an internal gear than its mating external gear, and
// the profile shift on the ring gear must be at least as big as the profile shift on the mating gear.
// You may need to use {{auto_profile_shift()}} to find this value if your mating gear has a small number of teeth.
// The gear spacing is given by {{gear_dist()}}.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// teeth = Total number of teeth around the spur gear.
// backing = The width of the ring gear backing. Default: height of teeth
// ---
// od = outer diameter of the ring
// or = outer radius of the ring
// width = width of the ring, measuring from tips of teeth to outside of ring.
// helical = The angle of the rack teeth away from perpendicular to the gear axis of rotation. Stretches out the tooth shapes. Used to match helical spur gear pinions. Default: 0
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees.
// profile_shift = Profile shift factor x for tooth profile. Default: 0
// clearance = Gap between top of a tooth on one gear and bottom of valley on a meshing gear (in millimeters)
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// Anchor Types:
// pitch = anchor on the pitch circle (default)
// outside = outside edge of the gear
// Example(2D,Big): Meshing a ring gear with a spur gear
// circ_pitch=5; teeth1=50; teeth2=18;
// dist = gear_dist(circ_pitch=circ_pitch, teeth1, teeth2, internal1=true);
// ring_gear2d(circ_pitch=circ_pitch, teeth=teeth1);
// color("lightblue")back(dist)
// spur_gear2d(circ_pitch=circ_pitch, teeth=teeth2);
// Example(2D,Med,VPT=[-0.117844,-0.439102,-0.372203],VPR=[0,0,0],VPD=192.044): Meshing a ring gear with an auto-profile-shifted spur gear:
// teeth1=7; teeth2=15;
// ps1=undef; // Allow auto profile shifting for first gear
// ps2=auto_profile_shift(teeth=teeth1);
// mod=3;
// d = gear_dist(mod=mod, teeth1=teeth1, teeth2=teeth2, profile_shift1=ps1, profile_shift2=ps2, internal2=true);
// ring_gear2d(mod=mod, teeth=teeth2,profile_shift=ps2);
// color("lightblue") fwd(d)
// spur_gear2d(mod=mod, teeth=teeth1, profile_shift=ps1);
module ring_gear2d(
circ_pitch,
teeth,
backing,
pressure_angle,
helical,
profile_shift=0,
clearance,
backlash = 0.0,
or,od,width,
pitch,
diam_pitch,
mod,
atype="pitch",
gear_spin = 0,shorten=0,
anchor = CENTER,
spin = 0
) {
circ_pitch = _inherit_gear_pitch("ring_gear2d()",pitch, circ_pitch, diam_pitch, mod);
PA = _inherit_gear_pa(pressure_angle);
helical = _inherit_gear_helical(helical);
checks =
assert(in_list(atype,["outside","pitch"]))
assert(is_finite(profile_shift), "Profile shift for ring gears must be numerical")
assert(is_integer(teeth) && teeth>3)
assert(num_defined([backing,or,od,width])<=1, "Cannot define more than one of backing, or, od and width")
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(is_finite(helical) && abs(helical)<90)
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(gear_spin));
pr = pitch_radius(circ_pitch, teeth, helical=helical);
ar = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=true);
rr=_root_radius(circ_pitch, teeth, clearance, profile_shift=profile_shift, internal=true);
or = is_def(or) ?
assert(is_finite(or) && or>ar, "or is invalid or too small for teeth")
or
: is_def(od) ?
assert(is_finite(od) && od>2*ar, "od is invalid or too small for teeth")
od/2
: is_def(width) ?
assert(is_finite(width) && width>ar-rr, "width is invalid or too small for teeth")
rr+width
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
ar+backing
: 2*ar - rr; // default case
attachable(anchor,spin, two_d=true, r=atype=="pitch"?pr:or) {
zrot(gear_spin)
difference() {
circle(r=or);
spur_gear2d(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
helical = helical,
clearance = clearance,
backlash = backlash,shorten=shorten,
profile_shift = profile_shift,
internal = true
);
}
children();
}
}
// Function&Module: rack()
// Synopsis: Creates a straight or helical gear rack.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: rack2d(), spur_gear(), spur_gear2d(), bevel_gear()
// Usage: As a Module
// rack(pitch, teeth, thickness, [base|bottom=|width=], [helical=], [pressure_angle=], [backlash=], [clearance=]) [ATTACHMENTS];
// rack(mod=, teeth=, thickness=, [base=|bottom=|width=], [helical=], [pressure_angle=], [backlash]=, [clearance=]) [ATTACHMENTS];
// Usage: As a Function
// vnf = rack(pitch, teeth, thickness, [base|bottom=|width=], [helical=], [pressure_angle=], [backlash=], [clearance=]);
// vnf = rack(mod=, teeth=, thickness=, [base=|bottom=|width=], [helical=], [pressure_angle=], [backlash=], [clearance=]);
// Description:
// This is used to create a 3D rack, which is a linear bar with teeth that a gear can roll along.
// A rack can mesh with any gear that has the same `pitch` and `pressure_angle`. A helical rack meshes with a gear with the opposite
// helical angle.
// When called as a function, returns a 3D [VNF](vnf.scad) for the rack.
// When called as a module, creates a 3D rack shape.
// .
// By default the rack has a backing whose height is equal to the height of the teeth. You can specify a different backing size
// or you can specify the total width of the rack (from the bottom of the rack to tooth tips) or the
// bottom point of the rack, which is the distance from the pitch line to the bottom of the rack.
// .
// The rack appears oriented with
// its teeth pointed UP, so to mesh with gears in the XY plane, use `orient=BACK` or `orient=FWD` and apply any desired rotation.
// The pitch line of the rack is aligned with the x axis, the TOP anchors are at the tips of the teeth and the BOTTOM anchors at
// the bottom of the backing. Note that for helical racks the corner anchors still point at 45 degr angles.
// Arguments:
// pitch = The pitch, or distance between teeth centers along the rack. Matches up with circular pitch on a spur gear. Default: 5
// teeth = Total number of teeth along the rack. Default: 20
// thickness = Thickness of rack. Default: 5
// backing = Distance from bottom of rack to the roots of the rack's teeth. (Alternative to bottom or width.) Default: height of rack teeth
// ---
// bottom = Distance from rack's pitch line (the x-axis) to the bottom of the rack. (Alternative to backing or width)
// width = Distance from base of rack to tips of teeth (alternative to bottom and backing).
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// helical = The angle of the rack teeth away from perpendicular to the rack length. Used to match helical spur gear pinions. Default: 0
// herringbone = If true, and helical is set, creates a herringbone rack.
// profile_shift = Profile shift factor x. Default: 0
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Named Anchors:
// "root" = At the base of the teeth, at the center of rack.
// "root-left" = At the base of the teeth, at the left end of the rack.
// "root-right" = At the base of the teeth, at the right end of the rack.
// "root-back" = At the base of the teeth, at the back of the rack.
// "root-front" = At the base of the teeth, at the front of the rack.
// Example(NoScales,VPR=[60,0,325],VPD=130):
// rack(pitch=5, teeth=10, thickness=5);
// Example(NoScales,VPT=[0.317577,3.42688,7.83665],VPR=[27.7,0,359.8],VPD=139.921): Rack for Helical Gear
// rack(pitch=5, teeth=10, thickness=5, backing=5, helical=30);
// Example(NoScales): Metric Rack, oriented BACK to align with a gear in default orientation. With profile shifting set to zero the gears mesh at their pitch circles.
// rack(mod=2, teeth=10, thickness=5, bottom=5, pressure_angle=14.5,orient=BACK);
// color("red") spur_gear(mod=2, teeth=18, thickness=5, pressure_angle=14.5,anchor=FRONT,profile_shift=0);
// Example(NoScales): Orienting the rack to the right using {zrot()}. In this case the gear has automatic profile shifting so we must use {{gear_dist()}} to correctly position the gear.
// zrot(-90)rack(mod=2, teeth=6, thickness=5, bottom=5, pressure_angle=14.5,orient=BACK);
// color("red")
// right(gear_dist(mod=2,0,12,pressure_angle=14.5))
// spur_gear(mod=2, teeth=12, thickness=5, pressure_angle=14.5);
// Example(NoScales,Anim,VPT=[0,0,12],VPD=100,Frames=18): Rack and Pinion with helical teeth
// teeth1 = 16; teeth2 = 16;
// pitch = 5; thick = 5; helical = 30;
// pr = pitch_radius(pitch, teeth2, helical=helical);
// pos = 3*(1-2*abs($t-1/2))-1.5;
// right(pr*2*PI/teeth2*pos)
// rack(pitch, teeth1, thickness=thick, helical=helical);
// up(pr)
// spur_gear(
// pitch, teeth2,
// thickness = thick,
// helical = -helical,
// shaft_diam = 5,
// orient = BACK,
// gear_spin = 180-pos*360/teeth2);
// Example(NoAxes,VPT=[-7.10396,-9.70691,3.50121],VPR=[60.2,0,325],VPD=213.262): Skew axis helical gear and rack engagement.
// mod=5; teeth=8; helical1=17.5; helical2=22.5;
// d = gear_dist_skew(mod=mod, teeth, 0, helical1,helical2);
// rack(mod=mod, teeth=5, thickness=30, helical=helical2, orient=FWD);
// color("lightblue")
// yrot(-helical1-helical2) fwd(d)
// spur_gear(mod=mod, teeth=teeth, helical=helical1, gear_spin=180/teeth, thickness=30);
module rack(
pitch,
teeth,
thickness,
backing,
width, bottom,
pressure_angle,
backlash = 0.0,
clearance,
helical,
herringbone = false,
profile_shift = 0,
gear_travel = 0,
circ_pitch,
diam_pitch,
mod,
anchor = CENTER,
spin = 0,
orient = UP
) {
pitch = _inherit_gear_pitch("rack()",pitch, circ_pitch, diam_pitch, mod, warn=false);
PA = _inherit_gear_pa(pressure_angle);
helical = _inherit_gear_helical(helical);
thickness = _inherit_gear_thickness(thickness);
checks=
assert(is_integer(teeth) && teeth>0)
assert(is_finite(thickness) && thickness>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_bool(herringbone))
assert(is_finite(profile_shift))
assert(is_finite(gear_travel));
trans_pitch = pitch / cos(helical);
a = _adendum(pitch, profile_shift);
d = _dedendum(pitch, clearance, profile_shift);
bottom = is_def(bottom) ?
assert(is_finite(bottom) && bottom>d, "bottom is invalid or too small for teeth")
bottom
: is_def(width) ?
assert(is_finite(width) && width>a+d, "width is invalid or too small for teeth")
width - a
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
backing+d
: 2*d+a; // default case
l = teeth * trans_pitch;
anchors = [
named_anchor("root", [0,0,-d], BACK),
named_anchor("root-left", [-l/2,0,-d], LEFT),
named_anchor("root-right", [ l/2,0,-d], RIGHT),
named_anchor("root-front", [0,-thickness/2,-d], FWD),
named_anchor("root-back", [0, thickness/2,-d], BACK),
];
endfix = sin(helical)*thickness/2;
override = function(anchor)
anchor.z==1 ? [ [anchor.x*l/2-endfix*anchor.y,anchor.y*thickness/2,a], undef, undef]
: anchor.x!=0 ? [ [anchor.x*l/2-endfix*anchor.y,anchor.y*thickness/2,anchor.z*bottom], undef,undef]
: undef;
size = [l, thickness, 2*bottom];
attachable(anchor,spin,orient, size=size, anchors=anchors, override=override) {
right(gear_travel)
xrot(90) {
if (herringbone) {
zflip_copy()
skew(axz=-helical)
linear_extrude(height=thickness/2, center=false, convexity=teeth*2) {
rack2d(
pitch = pitch,
teeth = teeth,
bottom = bottom,
pressure_angle = PA,
backlash = backlash,
clearance = clearance,
helical = helical,
profile_shift = profile_shift
);
}
} else {
skew(axz=helical)
linear_extrude(height=thickness, center=true, convexity=teeth*2) {
rack2d(
pitch = pitch,
teeth = teeth,
bottom = bottom,
pressure_angle = PA,
backlash = backlash,
clearance = clearance,
helical = helical,
profile_shift = profile_shift
);
}
}
}
children();
}
}
function rack(
pitch,
teeth,
thickness,
backing, bottom, width,
pressure_angle,
backlash = 0.0,
clearance,
helical,
herringbone = false,
profile_shift = 0,
circ_pitch,
diam_pitch,
mod,
gear_travel = 0,
anchor = CENTER,
spin = 0,
orient = UP
) =
let(
pitch = _inherit_gear_pitch("rack()",pitch, circ_pitch, diam_pitch, mod, warn=false),
PA = _inherit_gear_pa(pressure_angle),
helical = _inherit_gear_helical(helical),
thickness = _inherit_gear_thickness(thickness)
)
assert(is_integer(teeth) && teeth>0)
assert(is_finite(thickness) && thickness>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_bool(herringbone))
assert(is_finite(profile_shift))
assert(is_finite(gear_travel))
let(
trans_pitch = pitch / cos(helical),
a = _adendum(pitch, profile_shift),
d = _dedendum(pitch, clearance, profile_shift),
bottom = is_def(bottom) ?
assert(is_finite(bottom) && bottom>d, "bottom is invalid or too small for teeth")
bottom
: is_def(width) ?
assert(is_finite(width) && width>a+d, "width is invalid or too small for teeth")
width - a
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
backing+d
: 2*d+a, // default case
l = teeth * trans_pitch,
path = rack2d(
pitch = pitch,
teeth = teeth,
bottom = bottom,
pressure_angle = PA,
backlash = backlash,
clearance = clearance,
helical = helical,
profile_shift = profile_shift
),
vnf = herringbone
? sweep(path, [
left(adj_ang_to_opp(thickness/2,helical)) *
back(thickness/2) * xrot(90),
xrot(90),
left(adj_ang_to_opp(thickness/2,helical)) *
fwd(thickness/2) * xrot(90),
], style="alt", orient=FWD)
: skew(axy=-helical, p=linear_sweep(path, height=thickness, anchor="origin", orient=FWD)),
out = right(gear_travel, p=vnf),
size = [l, thickness, 2*bottom],
anchors = [
named_anchor("tip", [0,0,a], BACK),
named_anchor("tip-left", [-l/2,0,a], LEFT),
named_anchor("tip-right", [ l/2,0,a], RIGHT),
named_anchor("tip-front", [0,-thickness/2,a], DOWN),
named_anchor("tip-back", [0, thickness/2,a], UP),
named_anchor("root", [0,0,-d], BACK),
named_anchor("root-left", [-l/2,0,-d], LEFT),
named_anchor("root-right", [ l/2,0,-d], RIGHT),
named_anchor("root-front", [0,-thickness/2,-d], DOWN),
named_anchor("root-back", [0, thickness/2,-d], UP),
]
) reorient(anchor,spin,orient, size=size, anchors=anchors, p=out);
// Function&Module: rack2d()
// Synopsis: Creates a 2D gear rack.
// SynTags: Geom, Path
// Topics: Gears, Parts
// See Also: rack(), spur_gear(), spur_gear2d(), bevel_gear()
// Usage: As a Module
// rack2d(pitch, teeth, [base|bottom=|width=], [pressure_angle=], [backlash=], [clearance=]) [ATTACHMENTS];
// rack2d(mod=, teeth=, [base=|bottom=|width=], [pressure_angle=], [backlash=], [clearance=]) [ATTACHMENTS];
// Usage: As a Function
// path = rack2d(pitch, teeth, [base|bottom=|width=], [pressure_angle=], [backlash=], [clearance=]);
// path = rack2d(mod=, teeth=, [base=|bottom=|width=], [pressure_angle=], [backlash=], [clearance=]);
// Description:
// Create a 2D rack, a linear bar with teeth that a gear can roll along.
// A rack can mesh with any spur gear or helical gear that has the same `pitch` and `pressure_angle`.
// When called as a function, returns a 2D path for the outline of the rack.
// When called as a module, creates a 2D rack shape.
// .
// By default the rack has a backing whose height is equal to the height of the teeth. You can specify a different backing size
// or you can specify the total width of the rack (from the bottom of the rack to tooth tips) or the
// bottom point of the rack, which is the distance from the pitch line to the bottom of the rack.
// .
// The rack appears with its pitch line on top of the x axis. The BACK anchor refers to the tips of the teeth and the FRONT
// anchor refers to the front of the backing. You can use named anchors to access the roots of the teeth.
// Arguments:
// pitch = The pitch, or distance between teeth centers along the rack. Matches up with circular pitch on a spur gear. Default: 5
// teeth = Total number of teeth along the rack
// backing = Distance from bottom of rack to the roots of the rack's teeth. (Alternative to bottom or width.) Default: height of rack teeth
// ---
// bottom = Distance from rack's pitch line (the x-axis) to the bottom of the rack. (Alternative to backing or width)
// width = Distance from base of rack to tips of teeth (alternative to bottom and backing).
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// helical = The angle of the rack teeth away from perpendicular to the rack length. Stretches out the tooth shapes. Used to match helical spur gear pinions. Default: 0
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees.
// profile_shift = Profile shift factor x for tooth shape. Default: 0
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// gear_travel = The distance the rack should be moved by linearly. Default: 0
// rounding = If true, rack tips and valleys are slightly rounded. Default: true
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// Named Anchors:
// "root" = At the height of the teeth, at the center of rack.
// "root-left" = At the height of the teeth, at the left end of the rack.
// "root-right" = At the height of the teeth, at the right end of the rack.
// Example(2D):
// rack2d(pitch=5, teeth=10);
// Example(2D): Called as a Function
// path = rack2d(pitch=8, teeth=8, pressure_angle=25);
// polygon(path);
function rack2d(
pitch,
teeth,
backing,
pressure_angle,
backlash = 0,
clearance,
helical,
profile_shift = 0,
circ_pitch,
diam_pitch,
mod,
width, bottom,
gear_travel = 0,
rounding = true,
anchor = CENTER,
spin = 0
) = let(
pitch = _inherit_gear_pitch("rack2d()",pitch, circ_pitch, diam_pitch, mod, warn=false),
PA = _inherit_gear_pa(pressure_angle),
helical = _inherit_gear_helical(helical),
mod = module_value(circ_pitch=pitch)
)
assert(is_integer(teeth) && teeth>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_finite(gear_travel))
assert(num_defined([width,backing,bottom])<=1, "Can define only one of width, backing and bottom")
let(
adendum = _adendum(pitch, profile_shift),
dedendum = _dedendum(pitch, clearance, profile_shift),
clear = default(clearance, 0.25 * mod),
bottom = is_def(bottom) ?
assert(is_finite(bottom) && bottom>dedendum, "bottom is invalid or too small for teeth")
bottom
: is_def(width) ?
assert(is_finite(width) && width>adendum+dedendum, "width is invalid or too small for teeth")
width - adendum
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
backing+dedendum
: 2*dedendum+adendum // default case
)
let(
trans_pitch = pitch / cos(helical),
trans_pa = atan(tan(PA)/cos(helical)),
tthick = trans_pitch/PI * (PI/2 + 2*profile_shift * tan(PA)) - backlash,
l = teeth * trans_pitch,
ax = ang_adj_to_opp(trans_pa, adendum),
dx = dedendum*tan(trans_pa),
poff = tthick/2,
tooth = [
[-trans_pitch/2, -dedendum],
if (rounding) each arc(n=4, r=clear, corner=[
[-trans_pitch/2, -dedendum],
[-poff-dx, -dedendum],
[-poff+ax, +adendum],
]) else [-poff-dx, -dedendum],
if (rounding) each arc(n=4, r=trans_pitch/16, corner=[
[-poff-dx, -dedendum],
[-poff+ax, +adendum],
[+poff-ax, +adendum],
]) else [-poff+ax, +adendum],
if (rounding) each arc(n=4, r=trans_pitch/16, corner=[
[-poff+ax, +adendum],
[+poff-ax, +adendum],
[+poff+dx, -dedendum],
]) else [+poff-ax, +adendum],
if (rounding) each arc(n=4, r=clear, corner=[
[+poff-ax, +adendum],
[+poff+dx, -dedendum],
[+trans_pitch/2, -dedendum],
]) else [+poff+dx, -dedendum],
[+trans_pitch/2, -dedendum],
],
path2 = [
for(m = xcopies(trans_pitch,n=teeth))
each apply(m,tooth)
],
path = right(gear_travel, p=[
[path2[0].x, -bottom],
each path2,
[last(path2).x, -bottom],
]),
size=[l,2*bottom],
anchors = [
named_anchor("root", [ 0,-dedendum,0], BACK),
named_anchor("root-left", [-l/2,-dedendum,0], LEFT),
named_anchor("root-right", [ l/2,-dedendum,0], RIGHT),
],
override = [
[[0,1] , [[0,adendum]]],
[[1,1] , [[l/2,adendum]]],
[[-1,1] , [[-l/2,adendum]]],
]
) reorient(anchor,spin, two_d=true, size=size, anchors=anchors, override=override, p=path);
module rack2d(
pitch,
teeth,
backing,
width, bottom,
pressure_angle,
backlash = 0,
clearance,
helical,
profile_shift = 0,
gear_travel = 0,
circ_pitch,
diam_pitch,
mod,
rounding = true,
anchor = CENTER,
spin = 0
) {
pitch = _inherit_gear_pitch("rack2d()",pitch, circ_pitch, diam_pitch, mod, warn=false);
PA = _inherit_gear_pa(pressure_angle);
helical = _inherit_gear_helical(helical);
checks =
assert(is_integer(teeth) && teeth>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(helical) && abs(helical)<90)
assert(is_finite(gear_travel))
assert(num_defined([width,backing,bottom])<=1, "Can define only one of width, backing and bottom");
trans_pitch = pitch / cos(helical);
a = _adendum(pitch, profile_shift);
d = _dedendum(pitch, clearance, profile_shift);
bottom = is_def(bottom) ?
assert(is_finite(bottom) && bottom>d, "bottom is invalid or too small for teeth")
bottom
: is_def(width) ?
assert(is_finite(width) && width>a+d, "width is invalid or too small for teeth")
width - a
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
backing+d
: 2*d+a; // default case
l = teeth * trans_pitch;
path = rack2d(
pitch = pitch,
teeth = teeth,
bottom=bottom,
pressure_angle = PA,
backlash = backlash,
clearance = clearance,
helical = helical,
rounding=rounding,
profile_shift= profile_shift
);
size = [l, 2*bottom];
anchors = [
named_anchor("root", [ 0,-d,0], BACK),
named_anchor("root-left", [-l/2,-d,0], LEFT),
named_anchor("root-right", [ l/2,-d,0], RIGHT),
];
override = [
[[0,1] , [[0,a]]],
[[1,1] , [[l/2,a]]],
[[-1,1] , [[-l/2,a]]],
];
attachable(anchor,spin, two_d=true, size=size, anchors=anchors, override=override) {
right(gear_travel) polygon(path);
children();
}
}
// Function&Module: crown_gear()
// Synopsis: Creates a crown gear that can mesh with a spur gear.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: rack(), rack2d(), spur_gear(), spur_gear2d(), bevel_pitch_angle(), bevel_gear()
// Usage: As a Module
// crown_gear(circ_pitch, teeth, backing, face_width, [pressure_angle=], [clearance=], [backlash=], [profile_shift=], [slices=]);
// crown_gear(diam_pitch=, teeth=, backing=, face_width=, [pressure_angle=], [clearance=], [backlash=], [profile_shift=], [slices=]);
// crown_gear(mod=, teeth=, backing=, face_width=, [pressure_angle=], [clearance=], [backlash=], [profile_shift=], [slices=]);
// Usage: As a Function
// vnf = crown_gear(circ_pitch, teeth, backing, face_width, [pressure_angle=], [clearance=], [backlash=], [profile_shift=], [slices=]);
// vnf = crown_gear(diam_pitch=, teeth=, backing=, face_width=, [pressure_angle=], [clearance=], [backlash=], [profile_shift=], [slices=]);
// vnf = crown_gear(mod=, teeth=, backing=, face_width=, [pressure_angle=], [clearance=], [backlash=], [profile_shift=], [slices=]);
// Description:
// Creates a crown gear. The module `crown_gear()` gives a crown gear, with reasonable defaults
// for all the parameters. Normally, you should just choose the first 4 parameters, and let the
// rest be default values.
// .
// The module `crown_gear()` gives a crown gear in the XY plane, centered on the origin, with one tooth
// centered on the positive Y axis. The crown gear will have the pitch circle of the teeth at Z=0 by default.
// The inner radius of the crown teeth can be calculated with the `pitch_radius()` function, and the outer
// radius of the teeth is `face_width=` more than that.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle. Default: 5
// teeth = Total number of teeth around the entire perimeter. Default: 20
// backing = Distance from base of crown gear to roots of teeth (alternative to bottom and thickness).
// face_width = Width of the toothed surface, from inside radius to outside. Default: 5
// ---
// bottom = Distance from crown's pitch plane (Z=0) to the bottom of the crown gear. (Alternative to backing or thickness)
// thickness = Distance from base of crown gear to tips of teeth (alternative to bottom and backing).
// pitch_angle = Angle of beveled gear face. Default: 45
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// slices = Number of vertical layers to divide gear into. Useful for refining gears with `spiral`. Default: 1
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Example:
// crown_gear(mod=1, teeth=40, backing=3, face_width=5, pressure_angle=20);
// Example:
// mod=1; cteeth=40; pteeth=17; backing=3; PA=20; face=5;
// cpr = pitch_radius(mod=mod, teeth=cteeth);
// ppr = pitch_radius(mod=mod, teeth=pteeth);
// crown_gear(mod=mod, teeth=cteeth, backing=backing,
// face_width=face, pressure_angle=PA);
// back(cpr+face/2)
// up(ppr)
// spur_gear(mod=mod, teeth=pteeth,
// pressure_angle=PA, thickness=face,
// orient=BACK, gear_spin=180/pteeth,
// profile_shift=0);
function crown_gear(
circ_pitch,
teeth,
backing,
face_width=5,
pressure_angle=20,
clearance,
backlash=0,
profile_shift=0,
slices=10,
bottom,
thickness,
diam_pitch,
pitch,
mod,
gear_spin=0,
anchor=CTR,
spin=0,
orient=UP
) = let(
pitch = _inherit_gear_pitch("crown_gear()", pitch, circ_pitch, diam_pitch, mod, warn=false),
PA = _inherit_gear_pa(pressure_angle)
)
assert(is_integer(teeth) && teeth>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(gear_spin))
assert(num_defined([thickness,backing,bottom])<=1, "Can define only one of thickness, backing and bottom")
let(
a = _adendum(pitch, profile_shift),
d = _dedendum(pitch, clearance, profile_shift),
bottom = is_def(bottom) ?
assert(is_finite(bottom) && bottom>d, "bottom is invalid or too small for teeth")
bottom
: is_def(thickness) ?
assert(is_finite(thickness) && thickness>a+d, "thickness is invalid or too small for teeth")
thickness - a
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
backing+d
: 2*d+a, // default case
mod = module_value(circ_pitch=pitch),
ir = mod * teeth / 2,
or = ir + face_width,
profiles = [
for (slice = [0:1:slices-1])
let(
u = slice / (slices-1),
r = or - u*face_width,
wpa = acos(ir * cos(PA) / r),
profile = select(
rack2d(
mod=mod, teeth=1,
pressure_angle=wpa,
clearance=clearance,
backlash=backlash,
profile_shift=profile_shift,
rounding=false
), 2, -3
),
delta = profile[1] - profile[0],
slope = delta.y / delta.x,
C = profile[0].y - slope * profile[0].x,
profile2 = profile[1].x > 0
? [profile[0], [0,C], [0,C], profile[3]]
: profile,
m = back(r) * xrot(90),
tooth = apply(m, path3d(profile2)),
rpitch = pitch * r / ir
)
assert(profile[3].x <= rpitch/2, "face_width is too wide for the given gear geometry. Either decrease face_width, or increase the module or tooth count.")
[
for (i = [0:1:teeth-1])
let(a = gear_spin - i * 360 / teeth)
each zrot(a, p=tooth)
]
],
rows = [
[for (p=profiles[0]) [p.x,p.y,-bottom]],
each profiles,
[for (p=last(profiles)) [p.x,p.y,last(profiles)[0].z]],
],
vnf = vnf_vertex_array(rows, col_wrap=true, caps=true)
) reorient(anchor,spin,orient, r=or, h=2*bottom, p=vnf);
module crown_gear(
circ_pitch,
teeth,
backing,
face_width=10,
pressure_angle=20,
clearance,
backlash=0,
profile_shift=0,
slices=10,
bottom,
thickness,
diam_pitch,
pitch,
mod,
gear_spin=0,
anchor=CTR,
spin=0,
orient=UP
) {
pitch = _inherit_gear_pitch("crown_gear()", pitch, circ_pitch, diam_pitch, mod, warn=false);
PA = _inherit_gear_pa(pressure_angle);
checks =
assert(is_integer(teeth) && teeth>0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(gear_spin))
assert(num_defined([thickness,backing,bottom])<=1, "Can define only one of width, backing and bottom")
;
pr = pitch_radius(circ_pitch=pitch, teeth=teeth);
a = _adendum(pitch, profile_shift);
d = _dedendum(pitch, clearance, profile_shift);
bottom = is_def(bottom) ?
assert(is_finite(bottom) && bottom>d, "bottom is invalid or too small for teeth")
bottom
: is_def(thickness) ?
assert(is_finite(thickness) && thickness>a+d, "thickness is invalid or too small for teeth")
thickness - a
: is_def(backing) ?
assert(all_positive([backing]), "backing must be a positive value")
backing+d
: 2*d+a; // default case
vnf = crown_gear(
circ_pitch=pitch,
teeth=teeth,
bottom=bottom,
face_width=face_width,
pressure_angle=PA,
clearance=clearance,
backlash=backlash,
profile_shift=profile_shift,
slices=slices,
gear_spin=gear_spin
);
attachable(anchor,spin,orient, r=pr+face_width, h=2*bottom) {
vnf_polyhedron(vnf, convexity=teeth/2);
children();
}
}
// Function&Module: bevel_gear()
// Synopsis: Creates a straight, zerol, or spiral bevel gear.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: rack(), rack2d(), spur_gear(), spur_gear2d(), bevel_pitch_angle(), bevel_gear()
// Usage: As a Module
// gear_dist(mod=|diam_pitch=|circ_pitch=, teeth, mate_teeth, [shaft_angle], [shaft_diam], [face_width=], [hide=], [spiral=], [cutter_radius=], [right_handed=], [pressure_angle=], [backing=|thickness=|bottom=], [cone_backing=], [backlash=], [slices=], [internal=], [gear_spin=], ...) [ATTACHMENTS];
// Usage: As a Function
// vnf = gear_dist(mod=|diam_pitch=|circ_pitch=, teeth, mate_teeth, [shaft_angle], [face_width=], [hide=], [spiral=], [cutter_radius=], [right_handed=], [pressure_angle=], , [backing=|thickness=|bottom=], [cone_backing=], [backlash=], [slices=], [internal=], [gear_spin=], ...);
// Description:
// Creates a spiral, zerol, or straight bevel gear. In straight bevel gear sets, when each tooth
// engages it inpacts the corresponding tooth. The abrupt tooth engagement causes impact stress
// which makes them more prone to breakage. Spiral bevel gears have teeth formed along spirals so
// they engage more gradually, resulting in a less abrupt transfer of force, so they are quieter
// in operation and less likely to break.
// .
// Bevel gears must be created in mated pairs to work together at a chosen shaft angle. You therefore
// must specify both the number of teeth on the gear and the number of teeth on its mating gear.
// Additional requirements for bevel gears to mesh are that they share the same
// tooth size and the same pressure angle and they must be of opposite handedness.
// The pressure angle controls how much the teeth bulge at their
// sides and is almost always 20 degrees for standard bevel gears. The ratio of `teeth` for two meshing gears
// gives how many times one will make a full
// revolution when the the other makes one full revolution. If the two numbers are coprime (i.e.
// are not both divisible by the same number greater than 1), then every tooth on one gear will meet
// every tooth on the other, for more even wear. So relatively prime numbers of teeth are good.
// .
// The gear appears centered on the origin, with one tooth
// centered on the positive Y axis. The base of the pitch cone (the "pitchbase") will lie in the XY plane. This is
// the natural position: in order to mesh the mating gear must be positioned so their pitch bases are tangent.
// The apexes of the pitch cones must coincide.
// .
// By default backing will be added to ensure
// that the center of the gear (where there are no teeth) is at least half the face width in thickness.
// You can change this using the `backing`, `thickness` or `bottom` parameters. The backing appears with
// a conical shape, extended the sloped edges of the teeth. You can set `cone_backing=false` if your application
// requires cylindrical backing.
// Arguments:
// teeth = Number of teeth on the gear
// mate_teeth = Number of teeth on the gear that will mate to this gear
// shaft_angle = Angle between the shafts of the two gears. Default: 90
// --
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// backing = Distance from bottom of bevel gear to bottom corner of teeth (Alternative to bottom or thickness). Default: 0 if the gear is thick enough (see above)
// bottom = Distance from bevel gear's pitch base to the bottom of the bevel gear. (Alternative to backing or thickness)
// thickness = Thickness of bevel gear at the center, where there are no teeth. (Alternative to backing or bottom).
// cone_backing = If true backing extends conical shape of the gear; otherwise backing is an attached cylinder. Default: true
// face_width = Width of teeth. Default: minimum of one third the cone distance and 10*module
// shaft_diam = Diameter of the hole in the center, or zero for no hole. (Module only.) Default: 0
// hide = Number of teeth to delete to make this only a fraction of a circle. Default: 0
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// spiral = The base angle for spiral teeth. If zero the teeth will be zerol or straight. Default: 35
// cutter_radius = Radius of spiral arc for teeth. If 0, then gear will have straight teeth. Default: face_width/2/cos(spiral)
// right_handed = If true, the gear returned will have a right-handed teeth. Default: false
// slices = Number of vertical layers to divide gear into. Useful for refining gears with `spiral`. Default: 1
// gear_spin = Rotate gear and children around the gear center, regardless of how gear is anchored. Default: 0
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: "pitchbase"
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Named Anchors:
// "pitchbase" = With the base of the pitch cone in the XY plane, centered at the origin. This is the natural height for the gear, and the default anchor.
// "apex" = At the pitch cone apex for the bevel gear.
// "flattop" = At the top of the flat top of the bevel gear.
// Example(NoAxes): Bevel Gear with zerol teeth
// bevel_gear(
// circ_pitch=5, teeth=36, mate_teeth=36,
// shaft_diam=5, spiral=0
// );
// Example(NoAxes): Spiral Beveled Gear and Pinion. Note conical backing added to the yellow gear to prevent it from being thin.
// t1 = 16; t2 = 28;
// color("lightblue")bevel_gear(
// circ_pitch=5, teeth=t1, mate_teeth=t2,
// slices=12, anchor="apex", orient=FWD
// );
// bevel_gear(
// circ_pitch=5, teeth=t2, mate_teeth=t1, right_handed=true,
// slices=12, anchor="apex", backing=3, spin=180/t2
// );
// Example(Anim,Frames=4,VPD=175,NoAxes): Manual Spacing of Pinion and Gear. Here conical backing has been turned off.
// t1 = 14; t2 = 28; circ_pitch=5;
// color("lightblue")back(pitch_radius(circ_pitch, t2)) {
// yrot($t*360/t1)
// bevel_gear(
// circ_pitch=circ_pitch, teeth=t1, mate_teeth=t2, shaft_diam=5,
// slices=12, orient=FWD
// );
// }
// down(pitch_radius(circ_pitch, t1)) {
// zrot($t*360/t2)
// bevel_gear(
// circ_pitch=circ_pitch, teeth=t2, mate_teeth=t1, right_handed=true,
// shaft_diam=5, slices=12, backing=3, spin=180/t2, cone_backing=false
// );
// }
// Example(NoAxes,VPT=[24.4306,-9.20912,-29.3331],VPD=292.705,VPR=[71.8,0,62.5]): Bevel gears at a non right angle, positioned by aligning the pitch cone apexes.
// ang=65;
// bevel_gear(mod=3,35,15,ang,spiral=0,backing=5,anchor="apex")
// cyl(h=48,d=3,$fn=16,anchor=BOT);
// color("lightblue")
// xrot(ang)
// bevel_gear(mod=3,15,35,ang,spiral=0,right_handed=true,anchor="apex")
// cyl(h=65,d=3,$fn=16,anchor=BOT);
// Example(VPT=[6.39483,26.2195,8.93229],VPD=192.044,VPR=[76.7,0,63.3],NoAxes): At this extreme 135 degree angle the yellow gear has internal teeth. This is a rare configuration.
// ang=135;
// bevel_gear(mod=3,35,15,ang);
// color("lightblue")
// back(pitch_radius(mod=3,teeth=35)+pitch_radius(mod=3,teeth=15))
// xrot(ang,cp=[0,-pitch_radius(mod=3,teeth=15),0])
// bevel_gear(mod=3,15,35,ang,right_handed=true);
function bevel_gear(
teeth,
mate_teeth,
shaft_angle=90,
backing,thickness,bottom,
face_width,
pressure_angle = 20,
clearance,
backlash = 0.0,
cutter_radius,
spiral = 35,
right_handed = false,
slices = 5,
cone_backing = true,
pitch,
circ_pitch,
diam_pitch,
mod,
anchor = "pitchbase",
spin = 0,
gear_spin = 0,
orient = UP,
_return_anchors = false
) = assert(all_integer([teeth,mate_teeth]) && teeth>=3 && mate_teeth>=3, "Must give teeth and mate_teeth, integers greater than or equal to 3")
assert(all_nonnegative([spiral]), "spiral must be nonnegative")
assert(is_undef(cutter_radius) || all_nonnegative([cutter_radius]), "cutter_radius must be nonnegative")
assert(is_finite(shaft_angle) && shaft_angle>0 && shaft_angle<180,"shaft_angle must be strictly between 0 and 180")
let(
circ_pitch = _inherit_gear_pitch("bevel_gear()",pitch, circ_pitch, diam_pitch, mod),
PA = _inherit_gear_pa(pressure_angle),
spiral = _inherit_gear_helical(spiral),
slices = cutter_radius==0? 1 : slices,
pitch_angle = posmod(atan(sin(shaft_angle)/((mate_teeth/teeth)+cos(shaft_angle))),180),
pr = pitch_radius(circ_pitch, teeth),
rr = _root_radius(circ_pitch, teeth, clearance),
pitchoff = (pr-rr) * sin(pitch_angle),
ocone_rad = pitch_angle<90 ? opp_ang_to_hyp(pr, pitch_angle)
: opp_ang_to_hyp(pitch_radius(circ_pitch,mate_teeth), shaft_angle-pitch_angle),
default_face_width = min(ocone_rad/3, 10*module_value(circ_pitch)),
face_width = _inherit_gear_thickness(face_width,dflt=default_face_width),
icone_rad = ocone_rad - face_width,
cutter_radius = is_undef(cutter_radius) ? face_width * 2 / cos(spiral)
: cutter_radius==0? face_width*100
: cutter_radius,
midpr = (icone_rad + ocone_rad) / 2,
radcp = [0, midpr] + polar_to_xy(cutter_radius, 180+spiral),
angC1 = law_of_cosines(a=cutter_radius, b=norm(radcp), c=ocone_rad),
angC2 = law_of_cosines(a=cutter_radius, b=norm(radcp), c=icone_rad),
radcpang = v_theta(radcp),
sang = radcpang - (180-angC1),
eang = radcpang - (180-angC2),
profile = reverse(_gear_tooth_profile(
circ_pitch = circ_pitch,
teeth = teeth,
pressure_angle = PA,
clearance = clearance,
backlash = backlash,
center = true
)),
verts1 = [
for (v = lerpn(0,1,slices+1)) let(
p = radcp + polar_to_xy(cutter_radius, lerp(sang,eang,v)),
ang = v_theta(p)-90,
dist = norm(p)
) [
let(
u = dist / ocone_rad,
m = up((1-u) * pr / tan(pitch_angle)) *
up(pitchoff) *
zrot(ang/sin(pitch_angle)) *
back(u * pr) *
xrot(pitch_angle) *
scale(u)
)
for (tooth=[0:1:teeth-1])
each apply(xflip() * zrot(360*tooth/teeth) * m, path3d(profile))
]
],
botz = verts1[0][0].z, // bottom of center
topz = last(verts1)[0].z, // top of center
ctr_thickness = topz - botz,
vertices = [for (x=verts1) reverse(x)],
sides_vnf = vnf_vertex_array(vertices, caps=false, col_wrap=true, reverse=true),
top_verts = last(vertices),
bot_verts = vertices[0],
gear_pts = len(top_verts),
face_pts = gear_pts / teeth,
minbacking = -min(0,ctr_thickness),
backing = is_def(backing) ?
assert(all_nonnegative([backing]), "backing must be a non-negative value")
assert(ctr_thickness>0 || backing>0, "internal gears require backing>0")
backing-min(0,ctr_thickness)
: is_def(thickness) ?
let(thick_OK=is_finite(thickness) && (thickness>abs(ctr_thickness) || (thickness==ctr_thickness && ctr_thickness>0)))
assert(thick_OK, str("thickness is invalid or too small for teeth; thickness must be larger than ",abs(ctr_thickness)))
thickness-ctr_thickness
: is_def(bottom)?
assert(is_finite(bottom) && bottom-pitchoff>minbacking,
str("bottom is invalid or too small for teeth, must exceed ",minbacking+pitchoff))
bottom-pitchoff
: ctr_thickness>face_width/2 ? 0
: -ctr_thickness+face_width/2,
cpz = (topz + botz - backing) / 2,
teeth_top_faces =[
for (i=[0:1:teeth-1], j=[0:1:(face_pts/2)-1]) each [
[i*face_pts+j, (i+1)*face_pts-j-1, (i+1)*face_pts-j-2],
[i*face_pts+j, (i+1)*face_pts-j-2, i*face_pts+j+1]
]
],
flat_top_faces = [
for (i=[0:1:teeth-1]) each [
[gear_pts, (i+1)*face_pts-1, i*face_pts],
[gear_pts, ((i+1)%teeth)*face_pts, (i+1)*face_pts-1]
]
],
backing_vert = backing==0? []
: !cone_backing ? down(backing,[for(i=[0:1:teeth-1]) each( [bot_verts[i*face_pts], bot_verts[(i+1)*face_pts-1]])])
: let(
factor = tan(pitch_angle-90)*backing
)
[for(i=[0:1:teeth-1]) let(
A = bot_verts[i*face_pts],
B = bot_verts[(i+1)*face_pts-1],
adjA = point3d(factor*unit(point2d(A)),-backing),
adjB = point3d(factor*unit(point2d(B)),-backing)
)
each [ A+adjA, B+adjB]],
shift = len(bot_verts),
backing_bot_faces = backing==0? flat_top_faces
:[for (i=idx(backing_vert))
[shift+len(backing_vert), shift+(i+1)%len(backing_vert),shift+i]
],
backing_side_faces = backing==0 ? []
: [
for (i=[0:1:teeth-1])
each [
[shift+2*i,shift+(2*i+1),(i+1)*face_pts-1],
[shift+2*i+1,shift+2*((i+1)%teeth), ((i+1)%teeth)*face_pts],
[(i+1)*face_pts-1, i*face_pts, shift+2*i],
[((i+1)%teeth)*face_pts, (i+1)*face_pts-1, shift+2*i+1]
]
],
vnf1 = vnf_join([
[
[each top_verts, [0,0,top_verts[0].z]],
concat(teeth_top_faces, flat_top_faces)
],
[
[each bot_verts,each backing_vert, [0,0,bot_verts[0].z-backing] ],
[for (x=concat(teeth_top_faces,backing_bot_faces,backing_side_faces)) reverse(x)]
],
sides_vnf
]),
lvnf = right_handed? vnf1 : xflip(p=vnf1),
vnf = zrot(gear_spin,down(cpz, p=lvnf)),
anchors = [
named_anchor("pitchbase", [0,0,pitchoff-ctr_thickness/2+backing/2]),
named_anchor("flattop", [0,0,ctr_thickness/2+backing/2]),
named_anchor("apex", [0,0,hyp_ang_to_opp(pitch_angle<90?ocone_rad:icone_rad,90-pitch_angle)+pitchoff-ctr_thickness/2+backing/2])
],
final_vnf = reorient(anchor,spin,orient, vnf=vnf, extent=true, anchors=anchors, p=vnf)
)
_return_anchors==false ? final_vnf
: [final_vnf, anchors, ctr_thickness+backing];
module bevel_gear(
teeth,
mate_teeth,
shaft_angle=90,
bottom,backing,thickness,cone_backing=true,
face_width,
shaft_diam = 0,
pressure_angle = 20,
clearance = undef,
backlash = 0.0,
cutter_radius,
spiral = 35,
right_handed = false,
slices = 5,
pitch,
diam_pitch,
circ_pitch,
mod,
anchor = "pitchbase",
spin = 0,
gear_spin=0,
orient = UP
) {
vnf_anchors = bevel_gear(
circ_pitch = circ_pitch, mod=mod, diam_pitch=diam_pitch,
teeth = teeth,
mate_teeth = mate_teeth,
shaft_angle=shaft_angle,
bottom=bottom,thickness=thickness,backing=backing,cone_backing=cone_backing,
face_width = face_width,
pressure_angle = pressure_angle,
clearance = clearance,
backlash = backlash,
cutter_radius = cutter_radius,
spiral = spiral,
right_handed = right_handed,
slices = slices,
anchor=CENTER,
gear_spin=gear_spin,
_return_anchors=true
);
vnf=vnf_anchors[0];
anchors=vnf_anchors[1];
thickness = vnf_anchors[2];
attachable(anchor,spin,orient, vnf=vnf, extent=true, anchors=anchors) {
difference() {
vnf_polyhedron(vnf, convexity=teeth/2);
if (shaft_diam > 0)
cylinder(h=2*thickness, r=shaft_diam/2, center=true, $fn=max(12,segs(shaft_diam/2)));
}
children();
}
}
// Function&Module: worm()
// Synopsis: Creates a worm that will mate with a worm gear.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), rack(), rack2d(), spur_gear(), spur_gear2d(), bevel_pitch_angle(), bevel_gear()
// Usage: As a Module
// worm(circ_pitch, d, l, [starts=], [left_handed=], [pressure_angle=], [backlash=], [clearance=]);
// worm(mod=, d=, l=, [starts=], [left_handed=], [pressure_angle=], [backlash=], [clearance=]);
// Usage: As a Function
// vnf = worm(circ_pitch, d, l, [starts=], [left_handed=], [pressure_angle=], [backlash=], [clearance=]);
// vnf = worm(mod=, d=, l=, [starts=], [left_handed=], [pressure_angle=], [backlash=], [clearance=]);
// Description:
// Creates a worm shape that can be matched to a worm gear.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle. Default: 5
// d = The diameter of the worm. Default: 30
// l = The length of the worm. Default: 100
// starts = The number of lead starts. Default: 1
// left_handed = If true, the gear returned will have a left-handed spiral. Default: false
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Example:
// worm(circ_pitch=8, d=30, l=50, $fn=72);
// Example: Multiple Starts.
// worm(circ_pitch=8, d=30, l=50, starts=3, $fn=72);
// Example: Left Handed
// worm(circ_pitch=8, d=30, l=50, starts=3, left_handed=true, $fn=72);
// Example: Called as Function
// vnf = worm(circ_pitch=8, d=35, l=50, starts=2, left_handed=true, pressure_angle=20, $fn=72);
// vnf_polyhedron(vnf);
function worm(
circ_pitch,
d=30, l=100,
starts=1,
left_handed=false,
pressure_angle,
backlash=0,
clearance,
diam_pitch,
mod,
pitch,
gear_spin=0,
anchor=CENTER,
spin=0,
orient=UP
) =
let(
circ_pitch = _inherit_gear_pitch("worm()", pitch, circ_pitch, diam_pitch, mod),
PA = _inherit_gear_pa(pressure_angle)
)
assert(is_integer(starts) && starts>0)
assert(is_finite(l) && l>0)
//assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_bool(left_handed))
assert(is_finite(gear_spin))
let(
helical = asin(starts * circ_pitch / PI / d),
trans_pitch = circ_pitch / cos(helical),
tooth = xflip(
p=select(rack2d(
pitch=circ_pitch,
teeth=1,
pressure_angle=PA,
clearance=clearance,
backlash=backlash,
helical=helical,
profile_shift=0
), 1, -2)
),
rack_profile = [
for (t = xcopies(trans_pitch, n=2*ceil(l/trans_pitch)+1))
each apply(t, tooth)
],
steps = max(36, segs(d/2)),
step = 360 / steps,
zsteps = ceil(l / trans_pitch / starts * steps),
zstep = l / zsteps,
profiles = [
for (j = [0:1:zsteps]) [
for (i = [0:1:steps-1]) let(
u = i / steps - 0.5,
ang = 360 * (1 - u) + 90,
z = j*zstep - l/2,
zoff = trans_pitch * starts * u,
h = lookup(z+zoff, rack_profile)
)
cylindrical_to_xyz(d/2+h, ang, z)
]
],
vnf1 = vnf_vertex_array(profiles, caps=true, col_wrap=true, style="alt"),
m = product([
zrot(gear_spin),
if (left_handed) xflip(),
]),
vnf = apply(m, vnf1)
) reorient(anchor,spin,orient, d=d, l=l, p=vnf);
module worm(
circ_pitch,
d=15, l=100,
starts=1,
left_handed=false,
pressure_angle,
backlash=0,
clearance,
pitch,
diam_pitch,
mod,
gear_spin=0,
anchor=CENTER,
spin=0,
orient=UP
) {
circ_pitch = _inherit_gear_pitch("worm()", pitch, circ_pitch, diam_pitch, mod);
PA = _inherit_gear_pa(pressure_angle);
checks =
assert(is_integer(starts) && starts>0)
assert(is_finite(l) && l>0)
//assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_bool(left_handed))
assert(is_finite(gear_spin));
helical = asin(starts * circ_pitch / PI / d);
trans_pitch = circ_pitch / cos(helical);
vnf = worm(
circ_pitch=circ_pitch,
starts=starts,
d=d, l=l,
left_handed=left_handed,
pressure_angle=PA,
backlash=backlash,
clearance=clearance,
mod=mod
);
attachable(anchor,spin,orient, d=d, l=l) {
zrot(gear_spin) vnf_polyhedron(vnf, convexity=ceil(l/trans_pitch)*2);
children();
}
}
// Function&Module: enveloping_worm()
// Synopsis: Creates a double-enveloping worm that will mate with a worm gear.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), rack(), rack2d(), spur_gear(), spur_gear2d(), bevel_pitch_angle(), bevel_gear()
// Usage: As a Module
// enveloping_worm(circ_pitch, mate_teeth, d, [left_handed=], [starts=], [arc=], [pressure_angle=]);
// enveloping_worm(mod=, mate_teeth=, d=, [left_handed=], [starts=], [arc=], [pressure_angle=]);
// enveloping_worm(diam_pitch=, mate_teeth=, d=, [left_handed=], [starts=], [arc=], [pressure_angle=]);
// Usage: As a Function
// vnf = enveloping_worm(circ_pitch, mate_teeth, d, [left_handed=], [starts=], [arc=], [pressure_angle=]);
// vnf = enveloping_worm(mod=, mate_teeth=, d=, [left_handed=], [starts=], [arc=], [pressure_angle=]);
// vnf = enveloping_worm(diam_pitch=, mate_teeth=, d=, [left_handed=], [starts=], [arc=], [pressure_angle=]);
// Description:
// Creates a double-enveloping worm shape that can be matched to a worm gear.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle. Default: 5
// mate_teeth = The number of teeth in the mated worm gear.
// d = The pitch diameter of the worm at its middle.
// left_handed = If true, the gear returned will have a left-handed spiral. Default: false
// ---
// starts = The number of lead starts. Default: 1
// arc = Arc angle of the mated worm gear to envelop. Default: `2 * pressure_angle`
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Example:
// enveloping_worm(circ_pitch=8, mate_teeth=45, d=30, $fn=72);
// Example: Multiple Starts.
// enveloping_worm(circ_pitch=8, mate_teeth=33, d=30, starts=3, $fn=72);
// Example: Left Handed
// enveloping_worm(circ_pitch=8, mate_teeth=33, d=30, starts=3, left_handed=true, $fn=72);
// Example: Called as Function
// vnf = enveloping_worm(circ_pitch=8, mate_teeth=37, d=35, starts=2, left_handed=true, pressure_angle=20, $fn=72);
// vnf_polyhedron(vnf);
function enveloping_worm(
circ_pitch,
mate_teeth,
d,
left_handed=false,
starts=1,
arc,
pressure_angle,
gear_spin=0,
rounding=true,
taper=true,
diam_pitch,
mod,
pitch,
anchor=CTR,
spin=0,
orient=UP
) =
let(
circ_pitch = _inherit_gear_pitch("worm_gear()", pitch, circ_pitch, diam_pitch, mod),
pressure_angle = _inherit_gear_pa(pressure_angle),
arc = default(arc, 2*pressure_angle)
)
assert(is_integer(mate_teeth) && mate_teeth>10)
assert(is_finite(d) && d>0)
assert(is_bool(left_handed))
assert(is_integer(starts) && starts>0)
assert(is_finite(arc) && arc>10 && arc<=2*pressure_angle)
assert(is_finite(gear_spin))
let(
hsteps = segs(d/2),
vsteps = hsteps,
helical = asin(starts * circ_pitch / PI / d),
pr = pitch_radius(circ_pitch, mate_teeth, helical=helical),
taper_table = taper
? [
[-180, 0],
[-arc/2, 0],
[-arc/2*0.85, 0.75],
[-arc/2*0.8, 0.93],
[-arc/2*0.75, 1],
[+arc/2*0.75, 1],
[+arc/2*0.8, 0.93],
[+arc/2*0.85, 0.75],
[+arc/2, 0],
[+180, 0],
]
: [
[-180, 0],
[-arc/2-0.00001, 0],
[-arc/2, 1],
[+arc/2, 1],
[+arc/2+0.00001, 0],
[+180, 0],
],
tarc = 360 / mate_teeth,
rteeth = quantup(ceil(mate_teeth*arc/360),2)+1+2*starts,
rack_path = select(
rack2d(
circ_pitch, rteeth,
pressure_angle=pressure_angle,
rounding=rounding, spin=90
),
1,-2
),
adendum = _adendum(circ_pitch, profile_shift=0),
m1 = yscale(360/(circ_pitch*mate_teeth)) * left(adendum),
rows = [
for (i = [0:1:hsteps-1]) let(
u = i / hsteps,
theta = (1-u) * 360,
m2 = back(circ_pitch*starts*u),
polars = [
for (p=apply(m1*m2, rack_path))
if(p.y>=-arc-tarc && p.y<=arc+tarc)
[pr+p.x*lookup(p.y,taper_table)+adendum, p.y]
],
rpolars = mirror([-1,1],p=polars)
) [
for (j = [0:1:vsteps-1]) let(
v = j / (vsteps-1),
phi = (v-0.5) * arc,
minor_r = lookup(phi, rpolars),
xy = [d/2+pr,0] + polar_to_xy(minor_r,180-phi),
xyz = xrot(90,p=point3d(xy))
) zrot(theta, p=xyz)
]
],
ys = column(flatten(rows),1),
miny = min(ys),
maxy = max(ys),
vnf1 = vnf_vertex_array(transpose(rows), col_wrap=true, caps=true),
m = product([
zrot(gear_spin),
if (!left_handed) xflip(),
zrot(90),
]),
vnf = apply(m, vnf1)
) reorient(anchor,spin,orient, d=d, l=maxy-miny, p=vnf);
module enveloping_worm(
circ_pitch,
mate_teeth,
d,
left_handed=false,
starts=1,
arc,
pressure_angle=20,
gear_spin=0,
rounding=true,
taper=true,
diam_pitch,
mod,
pitch,
anchor=CTR,
spin=0,
orient=UP
) {
vnf = enveloping_worm(
mate_teeth=mate_teeth,
d=d,
left_handed=left_handed,
starts=starts,
arc=arc,
pressure_angle=pressure_angle,
gear_spin=gear_spin,
rounding=rounding,
taper=taper,
circ_pitch=circ_pitch,
diam_pitch=diam_pitch,
mod=mod,
pitch=pitch
);
bounds = pointlist_bounds(vnf[0]);
delta = bounds[1] - bounds[0];
attachable(anchor,spin,orient, d=max(delta.x,delta.y), l=delta.z) {
vnf_polyhedron(vnf, convexity=mate_teeth);
children();
}
}
// Function&Module: worm_gear()
// Synopsis: Creates a worm gear that will mate with a worm.
// SynTags: Geom, VNF
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), rack(), rack2d(), spur_gear(), spur_gear2d(), bevel_pitch_angle(), bevel_gear()
// Usage: As a Module
// worm_gear(circ_pitch, teeth, worm_diam, [worm_starts=], [worm_arc=], [crowning=], [left_handed=], [pressure_angle=], [backlash=], [clearance=], [slices=], [shaft_diam=]) [ATTACHMENTS];
// worm_gear(mod=, teeth=, worm_diam=, [worm_starts=], [worm_arc=], [crowning=], [left_handed=], [pressure_angle=], [backlash=], [clearance=], [slices=], [shaft_diam=]) [ATTACHMENTS];
// Usage: As a Function
// vnf = worm_gear(circ_pitch, teeth, worm_diam, [worm_starts=], [worm_arc=], [crowning=], [left_handed=], [pressure_angle=], [backlash=], [clearance=], [slices=]);
// vnf = worm_gear(mod=, teeth=, worm_diam=, [worm_starts=], [worm_arc=], [crowning=], [left_handed=], [pressure_angle=], [backlash=], [clearance=], [slices=]);
// Description:
// Creates a worm gear to match with a worm.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle. Default: 5
// teeth = Total number of teeth along the rack. Default: 30
// worm_diam = The pitch diameter of the worm gear to match to. Default: 30
// worm_starts = The number of lead starts on the worm gear to match to. Default: 1
// worm_arc = The arc of the worm to mate with, in degrees. Default: 45 degrees
// crowning = The amount to oversize the virtual hobbing cutter used to make the teeth, to add a slight crowning to the teeth to make them fit the work easier. Default: 1
// left_handed = If true, the gear returned will have a left-handed spiral. Default: false
// pressure_angle = Controls how straight or bulged the tooth sides are. In degrees. Default: 20
// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle. Default: 0
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// profile_shift = Profile shift factor x. Default: "auto"
// slices = The number of vertical slices to refine the curve of the worm throat. Default: 10
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
// orient = Vector to rotate top towards, after spin. See [orient](attachments.scad#subsection-orient). Default: `UP`
// Example: Right-Handed
// worm_gear(circ_pitch=5, teeth=36, worm_diam=30, worm_starts=1);
// Example: Left-Handed
// worm_gear(circ_pitch=5, teeth=36, worm_diam=30, worm_starts=1, left_handed=true);
// Example: Multiple Starts
// worm_gear(circ_pitch=5, teeth=36, worm_diam=30, worm_starts=4);
// Example: Metric Worm Gear
// worm_gear(mod=2, teeth=32, worm_diam=30, worm_starts=1);
// Example(Anim,Frames=4,FrameMS=125,VPD=220,VPT=[-15,0,0]): Meshing Worm and Gear
// $fn=36;
// circ_pitch = 5; starts = 4;
// worm_diam = 30; worm_length = 50;
// gear_teeth=36;
// right(worm_diam/2)
// yrot($t*360/starts)
// worm(
// d=worm_diam,
// l=worm_length,
// circ_pitch=circ_pitch,
// starts=starts,
// orient=BACK);
// left(pitch_radius(circ_pitch, gear_teeth))
// zrot(-$t*360/gear_teeth)
// worm_gear(
// circ_pitch=circ_pitch,
// teeth=gear_teeth,
// worm_diam=worm_diam,
// worm_starts=starts);
// Example: Meshing Worm and Gear Metricly
// $fn = 72;
// modulus = 2; starts = 3;
// worm_diam = 30; worm_length = 50;
// gear_teeth=36;
// right(worm_diam/2)
// worm(d=worm_diam, l=worm_length, mod=modulus, starts=starts, orient=BACK);
// left(pitch_radius(mod=modulus, teeth=gear_teeth))
// worm_gear(mod=modulus, teeth=gear_teeth, worm_diam=worm_diam, worm_starts=starts);
// Example: Called as Function
// vnf = worm_gear(circ_pitch=8, teeth=30, worm_diam=30, worm_starts=1);
// vnf_polyhedron(vnf);
function worm_gear(
circ_pitch,
teeth,
worm_diam,
worm_starts=1,
worm_arc=45,
crowning=0.1,
left_handed=false,
pressure_angle,
backlash=0,
clearance,
profile_shift="auto",
slices=10,
gear_spin=0,
pitch,
diam_pitch,
mod,
get_thickness=false,
anchor=CTR,
spin=0,
orient=UP
) =
let(
circ_pitch = _inherit_gear_pitch("worm_gear()", pitch, circ_pitch, diam_pitch, mod),
PA = _inherit_gear_pa(pressure_angle),
profile_shift = auto_profile_shift(teeth,PA,profile_shift=profile_shift)
)
assert(is_finite(worm_diam) && worm_diam>0)
assert(is_integer(teeth) && teeth>7)
assert(is_finite(worm_arc) && worm_arc>0 && worm_arc <= 60)
assert(is_integer(worm_starts) && worm_starts>0)
assert(is_bool(left_handed))
assert(is_finite(backlash))
assert(is_finite(crowning) && crowning>=0)
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(profile_shift))
let(
gear_arc = 2 * PA,
helical = asin(worm_starts * circ_pitch / PI / worm_diam),
full_tooth = apply(
zrot(90) * scale(0.99),
_gear_tooth_profile(
circ_pitch, teeth=teeth,
pressure_angle=PA,
profile_shift=-profile_shift,
clearance=clearance,
helical=helical,
center=true
)
),
ftl = len(full_tooth),
tooth_half1 = (select(full_tooth, 0, ftl/2-1)),
tooth_half2 = (select(full_tooth, ftl/2, -1)),
tang = 360 / teeth,
rteeth = quantdn(teeth * gear_arc / 360, 2) / 2 + 0.5,
pr = pitch_radius(circ_pitch, teeth, helical=helical),
oslices = slices * 4,
rows = [
for (data = [[tooth_half1,1], [tooth_half2,-1]])
let (
tooth_half = data[0],
dir = data[1]
)
for (pt = tooth_half) [
for (i = [0:1:oslices])
let (
u = i / oslices,
w_ang = worm_arc * (u - 0.5),
g_ang_delta = w_ang/360 * tang * worm_starts * (left_handed?1:-1),
m = zrot(dir*rteeth*tang+g_ang_delta, cp=[worm_diam/2+pr,0,0]) *
left(crowning) *
yrot(w_ang) *
right(worm_diam/2+crowning) *
zrot(-dir*rteeth*tang+g_ang_delta, cp=[pr,0,0]) *
xrot(180)
) apply(m, point3d(pt))
]
],
midrow = len(rows)/2,
goodcols = [
for (i = idx(rows[0]))
let(
p1 = rows[midrow-1][i],
p2 = rows[midrow][i]
)
if (p1.y > p2.y) i
],
dowarn = goodcols[0]==0? 0 : echo("Worm gear tooth arc reduced to fit."),
truncrows = [for (row = rows) [ for (i=goodcols) row[i] ] ],
zs = column(flatten(truncrows),2),
minz = min(zs),
maxz = max(zs),
zmax = max(abs(minz), abs(maxz))+0.05,
twang1 = v_theta(truncrows[0][0]),
twang2 = v_theta(last(truncrows[0])),
twang = modang(twang1 - twang2) / (maxz-minz),
resampled_rows = [for (row = truncrows) resample_path(row, n=slices, keep_corners=30, closed=false)],
tooth_rows = [
for (row = resampled_rows) [
zrot(twang*(zmax-row[0].z), p=[row[0].x, row[0].y, zmax]),
each row,
zrot(twang*(-zmax-last(row).z), p=[last(row).x, last(row).y, -zmax]),
],
]
)
get_thickness? zmax*2 :
let(
gear_rows = [
for (i = [0:1:teeth-1])
let(
m = zrot(i*tang) *
back(pr) *
zrot(-90) *
left(worm_diam/2)
)
for (row = tooth_rows)
apply(m, row)
],
vnf1 = vnf_vertex_array(transpose(gear_rows), col_wrap=true, caps=true),
vnf = apply(zrot(gear_spin), vnf1)
) reorient(anchor,spin,orient, r=pr, h=2*zmax, p=vnf);
module worm_gear(
circ_pitch,
teeth,
worm_diam,
worm_starts = 1,
worm_arc = 45,
crowning = 0.1,
left_handed = false,
pressure_angle,
backlash = 0,
clearance,
profile_shift="auto",
slices = 10,
shaft_diam = 0,
gear_spin=0,
pitch,
diam_pitch,
mod,
anchor = CENTER,
spin = 0,
orient = UP
) {
circ_pitch = _inherit_gear_pitch("worm_gear()", pitch, circ_pitch, diam_pitch, mod);
PA = _inherit_gear_pa(pressure_angle);
profile_shift = auto_profile_shift(teeth,PA,profile_shift=profile_shift);
checks =
assert(is_integer(teeth) && teeth>10)
assert(is_finite(worm_diam) && worm_diam>0)
assert(is_integer(worm_starts) && worm_starts>0)
assert(is_finite(worm_arc) && worm_arc>0 && worm_arc<90)
assert(is_finite(crowning) && crowning>=0)
assert(is_bool(left_handed))
assert(is_finite(PA) && PA>=0 && PA<90, "Bad pressure_angle value.")
assert(clearance==undef || (is_finite(clearance) && clearance>=0))
assert(is_finite(backlash) && backlash>=0)
assert(is_finite(shaft_diam) && shaft_diam>=0)
assert(slices==undef || (is_integer(slices) && slices>0))
assert(is_finite(profile_shift) && abs(profile_shift)<1)
assert(is_finite(gear_spin));
helical = asin(worm_starts * circ_pitch / PI / worm_diam);
pr = pitch_radius(circ_pitch, teeth, helical);
vnf = worm_gear(
circ_pitch = circ_pitch,
teeth = teeth,
worm_diam = worm_diam,
worm_starts = worm_starts,
worm_arc = worm_arc,
crowning = crowning,
left_handed = left_handed,
pressure_angle = PA,
backlash = backlash,
clearance = clearance,
profile_shift = profile_shift,
slices = slices
);
thickness = pointlist_bounds(vnf[0])[1].z;
attachable(anchor,spin,orient, r=pr, l=thickness) {
zrot(gear_spin)
difference() {
vnf_polyhedron(vnf, convexity=teeth/2);
if (shaft_diam > 0) {
cylinder(h=2*thickness+1, r=shaft_diam/2, center=true, $fn=max(12,segs(shaft_diam/2)));
}
}
children();
}
}
/// Function: _gear_tooth_profile()
/// Usage: As Function
/// path = _gear_tooth_profile(pitch, teeth, [pressure_angle], [clearance], [backlash], [internal]);
/// Topics: Gears
/// See Also: spur_gear2d()
/// Description:
/// When called as a function, returns the 2D profile path for an individual gear tooth.
/// When called as a module, creates the 2D profile shape for an individual gear tooth.
/// Arguments:
/// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
/// teeth = Total number of teeth on the spur gear that this is a tooth for.
/// pressure_angle = Pressure Angle. Controls how straight or bulged the tooth sides are. In degrees.
/// clearance = Gap between top of a tooth on one gear and bottom of valley on a meshing gear (in millimeters)
/// backlash = Gap between two meshing teeth, in the direction along the circumference of the pitch circle
/// internal = If true, create a mask for difference()ing from something else.
/// center = If true, centers the pitch circle of the tooth profile at the origin. Default: false.
/// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
/// mod = The module of the gear (pitch diameter / teeth)
/// Example(2D):
/// _gear_tooth_profile(circ_pitch=5, teeth=20, pressure_angle=20);
/// Example(2D): Metric Gear Tooth
/// _gear_tooth_profile(mod=2, teeth=20, pressure_angle=20);
/// Example(2D):
/// _gear_tooth_profile(
/// circ_pitch=5, teeth=20, pressure_angle=20
/// );
/// Example(2D): As a function
/// path = _gear_tooth_profile(
/// circ_pitch=5, teeth=20, pressure_angle=20
/// );
/// stroke(path, width=0.1);
function _gear_tooth_profile(
circ_pitch,
teeth,
pressure_angle = 20,
clearance,
backlash = 0.0,
helical = 0,
internal = false,
profile_shift = 0.0,
shorten = 0,
mod,
diam_pitch,
pitch,
center = false
) = let(
// Calculate a point on the involute curve, by angle.
_involute = function(base_r,a)
let(b=a*PI/180) base_r * [cos(a)+b*sin(a), sin(a)-b*cos(a)],
steps = 16,
circ_pitch = circular_pitch(pitch=pitch, circ_pitch=circ_pitch, diam_pitch=diam_pitch, mod=mod),
mod = module_value(circ_pitch=circ_pitch),
clear = default(clearance, 0.25 * mod),
// Calculate the important circle radii
arad = outer_radius(circ_pitch, teeth, helical=helical, profile_shift=profile_shift, internal=internal, shorten=shorten),
prad = pitch_radius(circ_pitch, teeth, helical=helical),
brad = _base_radius(circ_pitch, teeth, pressure_angle, helical=helical),
rrad = _root_radius(circ_pitch, teeth, clearance, helical=helical, profile_shift=profile_shift, internal=internal),
srad = max(rrad,brad),
tthick = circ_pitch/PI / cos(helical) * (PI/2 + 2*profile_shift * tan(pressure_angle)) + (internal?backlash:-backlash),
tang = tthick / prad / 2 * 180 / PI,
// Generate a lookup table for the involute curve angles, by radius
involute_lup = [
for (i=[0:5:arad/PI/brad*360])
let(
xy = _involute(brad,i),
pol = xy_to_polar(xy)
)
if (pol.x <= arad * 1.1)
[pol.x, 90-pol.y]
],
// Generate reverse lookup table for involute radii, by angle
involute_rlup = mirror([-1,1],p=involute_lup), // swaps X and Y columns.
a_ang = lookup(arad, involute_lup),
p_ang = lookup(prad, involute_lup),
b_ang = lookup(brad, involute_lup),
r_ang = lookup(rrad, involute_lup),
s_ang = lookup(srad, involute_lup),
soff = tang + (b_ang - p_ang),
ma_rad = min(arad, lookup(90-soff+0.05*360/teeth/2, involute_rlup)),
ma_ang = lookup(ma_rad, involute_lup),
cap_steps = ceil((ma_ang + soff - 90) / 5),
cap_step = (ma_ang + soff - 90) / cap_steps,
ax = circ_pitch/4 - ang_adj_to_opp(pressure_angle, circ_pitch/PI),
// Calculate the undercut a meshing rack might carve out of this tooth.
undercut = [
for (a=[atan2(ax,rrad):-1:-90])
let(
bx = -a/360 * 2*PI*prad,
x = bx + ax,
y = prad - circ_pitch/PI + profile_shift*circ_pitch/PI,
pol = xy_to_polar(x,y)
)
if (pol.x < arad*1.05)
[pol.x, pol.y-a+180/teeth]
],
uc_min = min_index(column(undercut,0)),
// Generate a fast lookup table for the undercut.
undercut_lup = [for (i=idx(undercut)) if (i>=uc_min) undercut[i]],
// The u values to use when generating the tooth.
us = [for (i=[0:1:steps*2]) i/steps/2],
// Find top of undercut.
undercut_max = max([
0,
for (u = us) let(
r = lerp(rrad, ma_rad, u),
a1 = lookup(r, involute_lup) + soff,
a2 = lookup(r, undercut_lup),
a = internal || r < undercut_lup[0].x? a1 : min(a1,a2),
b = internal || r < undercut_lup[0].x? false : a1>a2
) if(a<90+180/teeth && b) r
]),
// Generate the left half of the tooth.
tooth_half_raw = deduplicate([
for (u = us)
let(
r = lerp(rrad, ma_rad, u),
a1 = lookup(r, involute_lup) + soff,
a2 = lookup(r, undercut_lup),
a = internal || r < undercut_lup[0].x? a1 : min(a1,a2)
)
if ( internal || r > (rrad+clear) )
if (!internal || r < (ma_rad-clear) )
if (a < 90+180/teeth)
polar_to_xy(r, a),
if (!internal)
for (i=[0:1:cap_steps-1]) let(
a = ma_ang + soff - i * (cap_step-1)
) polar_to_xy(ma_rad, a),
]),
// Round out the clearance valley
rcircum = 2 * PI * (internal? ma_rad : rrad),
rpart = (180/teeth-tang)/360,
round_r = min(clear, rcircum*rpart),
line1 = internal
? select(tooth_half_raw,-2,-1)
: select(tooth_half_raw,0,1),
line2 = internal
? [[0,ma_rad],[-1,ma_rad]]
: zrot(180/teeth, p=[[0,rrad],[1,rrad]]),
isect_pt = line_intersection(line1,line2),
rcorner = internal
? [last(line1), isect_pt, line2[0]]
: [line2[0], isect_pt, line1[0]],
rounded_tooth_half = deduplicate([
if (!internal && round_r>0) each arc(n=8, r=round_r, corner=rcorner),
if (!internal && round_r<=0) isect_pt,
each tooth_half_raw,
if (internal && round_r>0) each arc(n=8, r=round_r, corner=rcorner),
if (internal && round_r<=0) isect,
]),
// Strip "jaggies" if found.
strip_left = function(path,i)
i > len(path)? [] :
norm(path[i]) >= undercut_max? [for (j=idx(path)) if(j>=i) path[j]] :
let(
angs = [
for (j=[i+1:1:len(path)-1]) let(
p = path[i],
np = path[j],
r = norm(np),
a = v_theta(np-p)
) if(r<undercut_max) a
],
mti = !angs? 0 : min_index(angs),
out = concat([path[i]], strip_left(path, i + mti + 1))
) out,
tooth_half = !undercut_max? rounded_tooth_half :
strip_left(rounded_tooth_half, 0),
// Mirror the tooth to complete it.
full_tooth = deduplicate([
each tooth_half,
each reverse(xflip(tooth_half)),
]),
// Reduce number of vertices.
tooth = path_merge_collinear(
resample_path(full_tooth, n=ceil(2*steps), keep_corners=30, closed=false)
),
out = center? fwd(prad, p=tooth) : tooth
) out;
// Section: Gear Assemblies
// Function: planetary_gears()
// Synopsis: Calculate teeth counts and angles for planetary gear assembly with specified ratio.
// Usage:
// gear_data = planetary_gears(mod=|circ_pitch=|diam_pitch=, n, max_teeth, ring_carrier=|carrier_ring=|sun_carrier=|carrier_sun=|sun_ring=|ring_sun=, [helical=], [gear_spin=]);
// Description:
// Calculates a planetary gear assembly that approximates a desired transmission ratio. A planetary gear assembly can be regarded as having three
// elements: the outer ring gear, the central sun gear, and a carrier that holds several planet gears, which fit between the sun and ring.
// The transmission ratio of a planetary gear assembly depends on which element is fixed and which ones are considered the input and output shafts.
// The fixed element can be the ring gear, the sun gear, or the carrier, and then you specify the desired ratio between the other two.
// You must also specify a maximum number of teeth on the ring gear. The function calculates the best approximation to your desired
// transmission ratio under that constraint: a large enough increase in the allowed number of teeth will yield a more accurate approximation. Note that the planet gears
// appear uniformly spaced around the sun gear, but this uniformity is often only approximate. Exact uniformity occurs when teeth_sun+teeth_ring
// is a multiple of the number of planet gears.
// .
// You specify the desired ratio using one of six parameters that identify which ratio you want to specify, and which is the driven element.
// Each different ratio is limited to certain bounds. For the case of the fixed carrier system, the sun and ring rotate in opposite directions.
// This is sometimes indicated by a negative transmission ratio. For these cases you can give a positive or negative value.
// .
// The return is a list of entries that describe the elements of the planetary assembly. The list entries are:
// - ["sun", teeth, profile_shift, spin]
// - ["ring", teeth, profile_shift, spin]
// - ["planets", teeth, profile_shift, spins, positions, angles]
// - ["ratio", realized_ratio]
// .
// The sun and ring gear are assumed to be placed at the origin. The planet gears are placed at the list of positions. The gears all
// have a spin in degrees. The planets list also includes the angular position of each planet in the `angles` list.
// One of the planets always appears on the X+ axis when `gear_spin` is zero. The final list entry gives the realized ratio of
// the assembly, so you can determine how closely it approaches your desired ratio. This will always be a positive value.
// .
// The sun gear appears by default with a tooth pointing on the Y+ axis with no spin, so if gear_spin is not used then the sun gear spin will
// always be zero. If you set `gear_spin` then the drive gear for the ratio you specified will be rotated by the specified angle and all
// of the other gears will be rotated appropriately.
// .
// The computation of planetary gear assembles is about determining the teeth counts on the sun, ring and planet gears,
// and the angular positions of the planet gears.
// The tooth size or helical angle are needed only for determining proper profile shifting and for determining the
// gear positions for the profiled shifted gears. To control the size of the assembly, do a planetary calculation
// with a module of 1 and then scale the module to produce the required gear dimensions. Remember, you should never
// use `scale()` on gears; change their size by scaling the module or one of the other tooth size parameters.
// Arguments:
// n = Number of planetary gears
// max_teeth = maximum number of teeth allowed on the ring gear
// ---
// mod = The module of the gear, pitch diameter divided by tooth count.
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// circ_pitch = distance between teeth centers around the pitch circle.
// ring_carrier = set ring/carrier ratio to this value in a ring driven system, must be between 1 and 2
// carrier_ring = set carrier/ring ratio to this value in a carrier driven system, must be between 1/2 and 1
// sun_carrier = set sun/carrier ratio to this value in a sun driven system, must be larger than 2
// carrier_sun = set carrier/sun ratio to this value in a carrier driven system, must be smaller than 1/2
// ring_sun = set ring/sun ratio to this value in a ring driven system, must have absolute value larger than 1
// sun_ring = set sun/ring ratio to this value in a sun driven system, must have absolute value smaller than 1
// helical = create gears with specified helical angle. Default: 0
// gear_spin = rotate the driven gear by this number of degrees. Default:0
// Example(2D,NoAxes,Anim,Frames=90,FrameMS=30,VPT=[-0.875705,-0.110537,-66.3877],VPR=[0,0,0],VPD=102,Med): In this example we request a ring/carrier ratio of 1.341 and the system produced has a ratio of 4/3. The sun is fixed, the input is carried by the ring, and the carrier, shown as the blue triangle, is the output, rotating approximately in accordance with the requested ratio.
// mod=1;
// gear_data = planetary_gears(mod=mod, n=3, max_teeth=28, ring_carrier=1.341, gear_spin=4/3*360/3*$t);
// ring_gear2d(mod=mod, teeth=gear_data[1][1], profile_shift=gear_data[1][2], gear_spin=gear_data[1][3],backing=2);
// stroke(gear_data[2][4],closed=true,color="blue",width=2);
// spur_gear2d(mod=mod, teeth=gear_data[0][1], profile_shift=gear_data[0][2], gear_spin=gear_data[0][3]); //sun
// color("red")move_copies(gear_data[2][4])
// spur_gear2d(mod=mod, teeth=gear_data[2][1], profile_shift=gear_data[2][2], gear_spin=gear_data[2][3][$idx]);
// Example(2D,Med,NoAxes,Anim,FrameMS=60,Frames=90,VPT=[-0.125033,0.508151,-66.3877],VPR=[0,0,0],VPD=192.044): In this example we request a sun/carrier ratio of 3.6 and get exactly that ratio. The carrier shown as the blue pentagon moves very slowly as the central sun turns. The ring is fixed.
// mod=1;
// gear_data = planetary_gears(mod=mod, n=5, max_teeth=70, sun_carrier=3.6, gear_spin=3.6*360/5*$t);
// ring_gear2d(mod=mod, teeth=gear_data[1][1], profile_shift=gear_data[1][2], gear_spin=gear_data[1][3],backing=2);
// stroke(gear_data[2][4],closed=true,color="blue");
// color("gold")
// spur_gear2d(mod=mod, teeth=gear_data[0][1], profile_shift=gear_data[0][2], gear_spin=gear_data[0][3]); //sun
// color("red")move_copies(gear_data[2][4])
// spur_gear2d(mod=mod, teeth=gear_data[2][1], profile_shift=gear_data[2][2], gear_spin=gear_data[2][3][$idx]);
// Example(3D,Med,NoAxes,Anim,Frames=7,FrameMS=50,VPT=[0.128673,0.24149,0.651451],VPR=[38.5,0,21],VPD=222.648): Here we request a sun/ring ratio of 3 and it is exactly achieved. The carrier, shown in blue, is fixed. This example is shown with helical gears. It is important to remember to flip the sign of the helical angle for the planet gears.
// $fn=81;
// mod=1;
// helical=25;
// gear_data = planetary_gears(mod=mod, n=4, max_teeth=82, sun_ring=3, helical=helical,gear_spin=360/27*$t);
// ring_gear(mod=mod, teeth=gear_data[1][1], profile_shift=gear_data[1][2], helical=helical, gear_spin=gear_data[1][3],backing=4,thickness=7);
// color("blue"){
// move_copies(gear_data[2][4]) cyl(h=12,d=4);
// down(9)linear_extrude(height=3)scale(1.2)polygon(gear_data[2][4]);
// }
// spur_gear(mod=mod, teeth=gear_data[0][1], profile_shift=gear_data[0][2], helical=helical, gear_spin=gear_data[0][3]); //sun
// color("red")move_copies(gear_data[2][4])
// spur_gear(mod=mod, teeth=gear_data[2][1], profile_shift=gear_data[2][2], helical=-helical, gear_spin=gear_data[2][3][$idx]);
function planetary_gears(n, max_teeth, helical=0, circ_pitch, mod, diam_pitch,
ring_carrier, carrier_ring, sun_carrier, carrier_sun, sun_ring, ring_sun,
gear_spin=0) =
let(
mod = module_value(mod=mod,circ_pitch=circ_pitch,diam_pitch=diam_pitch),
dummy = one_defined([ring_carrier,carrier_ring,sun_carrier,carrier_sun,sun_ring,ring_sun],
"ring_carrier,carrier_ring,sun_carrier,carrier_sun,sun_ring,ring_sun"),
// ratio is between the sun and ring
ratio = is_def(ring_carrier) ? assert(is_finite(ring_carrier) && ring_carrier>1 && ring_carrier<2, "ring/carrier ratio must be between 1 and 2")
ring_carrier - 1
: is_def(carrier_ring) ? assert(is_finite(carrier_ring) && carrier_ring>1/2 && carrier_ring<1, "carrier/ring ratio must be between 1/2 and 1")
1/carrier_ring - 1
: is_def(sun_carrier) ? assert(is_finite(sun_carrier) && sun_carrier>2, "sun/carrier ratio must be larger than 2")
1/(sun_carrier-1)
: is_def(carrier_sun) ? assert(is_finite(carrier_sun) && carrier_sun<1/2, "carrier/sun ratio must be smaller than 1/2")
1/(1/carrier_sun-1)
: is_def(sun_ring) ? assert(is_finite(sun_ring) && abs(sun_ring)>1, "abs(sun/ring) ratio must be larger than 1")
1/abs(sun_ring)
: /*is_def(ring_sun)*/ assert(is_finite(ring_sun) && abs(ring_sun)<1, "abs(ring/sun) ratio must be smaller than 1")
abs(ring_sun),
pq = rational_approx(ratio, max_teeth),
factor = floor(max_teeth/pq[1]),
temp_z_sun = factor*pq[0],
temp_z_ring = factor*pq[1],
z_sun = temp_z_sun%2==0 ? temp_z_sun+1 : temp_z_sun,
z_ring = temp_z_ring%2==0 ? min(temp_z_ring+1, max_teeth-(max_teeth%2==0?1:0)) : temp_z_ring,
z_planet = (z_ring-z_sun)/2
)
assert(z_planet==floor(z_planet),"Planets have non-integer teeth count! Algorithm failed.")
let(
d12 = gear_dist(mod=mod,z_sun,z_planet,helical),
ps_sun = auto_profile_shift(teeth=z_sun,helical=helical),
ps_planet = auto_profile_shift(teeth=z_planet,helical=helical),
ps_ring = ps_sun+2*ps_planet,
ring_spin = ring_sun || ring_carrier ? gear_spin
: sun_ring ? -gear_spin*z_sun/z_ring
: carrier_ring ? gear_spin*(z_ring+z_sun)/z_ring
: 0,
planet_rot = ring_carrier ? gear_spin*z_ring/(z_ring+z_sun)
: carrier_sun || carrier_ring ? gear_spin
: sun_carrier ? gear_spin*z_sun/(z_ring+z_sun)
: carrier_ring ? gear_spin*z_ring/(z_ring+z_sun)
: 0,
sun_spin = ring_sun ? -gear_spin*z_ring/z_sun
: sun_ring || sun_carrier ? gear_spin
: carrier_sun ? (z_ring+z_sun)*gear_spin/z_sun
: 0,
planet_spin = -sun_spin*z_sun/z_planet,
quant = 360/(z_sun+z_ring),
planet_angles = [for (uang=lerpn(0,360,n,endpoint=false)) quant(uang,quant)+planet_rot],
planet_pos = [for(ang=planet_angles) d12*[cos(ang),sin(ang)]],
planet_spins = [for(ang=planet_angles) (z_sun/z_planet)*(ang-90)+90+ang+360/z_planet/2+planet_spin],
final_ratio = ring_carrier ? 1+z_sun/z_ring
: carrier_ring ? 1/(1+z_sun/z_ring)
: sun_carrier ? 1+z_ring/z_sun
: carrier_sun ? 1/(1+z_ring/z_sun)
: sun_ring ? z_ring/z_sun
: /* ring_run */ z_sun/z_ring
)
[
["sun", z_sun, ps_sun, sun_spin],
["ring", z_ring, ps_ring, 360/z_ring/2 * (1-(z_sun%2))+ring_spin],
["planets", z_planet, ps_planet, planet_spins, planet_pos, planet_angles],
["ratio", final_ratio]
];
// Section: Computing Gear Dimensions
// These functions let the user find the derived dimensions of the gear.
// A gear fits within a circle of radius outer_radius, and two gears should have
// their centers separated by the sum of their pitch_radius.
// Function: circular_pitch()
// Synopsis: Returns tooth density expressed as "circular pitch".
// Topics: Gears, Parts
// See Also: spur_gear(), diametral_pitch(), circular_pitch(), module_value()
// Usage:
// circ_pitch = circular_pitch(circ_pitch);
// circ_pitch = circular_pitch(mod=);
// circ_pitch = circular_pitch(diam_pitch=);
// Description:
// Get tooth size expressed as "circular pitch", or the distance between teeth centers around the pitch circle.
// For example, an 11 tooth gear with a pitch circumference of 110 mm has a circular pitch of 110 mm /11, or 10 mm / tooth.
// Note that this calculation is does not depend on units for circ_pitch or mod, but the `diam_pitch` argument is based
// on inches and returns its value in millimeters.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// Example(2D,Med,VPT=[0,31,0],VPR=[0,0,0],VPD=40):
// $fn=144;
// teeth=20;
// circ_pitch = circular_pitch(diam_pitch=8);
// pr = pitch_radius(circ_pitch, teeth);
// stroke(spur_gear2d(circ_pitch, teeth), width=0.1);
// color("cyan")
// dashed_stroke(circle(r=pr), width=0.1);
// color("black") {
// stroke(
// arc(r=pr, start=90+90/teeth, angle=-360/teeth),
// width=0.2, endcaps="arrow");
// back(pr+1) right(3)
// zrot(30) text("Circular Pitch", size=1);
// }
// Example:
// circ_pitch1 = circular_pitch(circ_pitch=5);
// circ_pitch2 = circular_pitch(diam_pitch=12);
// circ_pitch3 = circular_pitch(mod=2);
function circular_pitch(circ_pitch, mod, pitch, diam_pitch) =
assert(one_defined([pitch, mod, circ_pitch, diam_pitch], "pitch,mod,circ_pitch,diam_pitch"))
pitch != undef? assert(is_finite(pitch) && pitch>0) pitch :
circ_pitch != undef? assert(is_finite(circ_pitch) && circ_pitch>0) circ_pitch :
diam_pitch != undef? assert(is_finite(diam_pitch) && diam_pitch>0) PI / diam_pitch * INCH :
assert(is_finite(mod) && mod>0) mod * PI;
// Function: diametral_pitch()
// Synopsis: Returns tooth density expressed as "diametral pitch".
// Topics: Gears, Parts
// See Also: spur_gear(), diametral_pitch(), circular_pitch(), module_value()
// Usage:
// dp = diametral_pitch(circ_pitch);
// dp = diametral_pitch(mod=);
// dp = diametral_pitch(diam_pitch=);
// Description:
// Returns tooth density expressed as "diametral pitch", the number of teeth per inch of pitch diameter.
// For example, if you have a gear with 30 teeth, with a 1.5 inch pitch diameter, then you have a
// diametral pitch of 20 teeth/inch.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// Example:
// diam_pitch1 = diametral_pitch(mod=2);
// diam_pitch2 = diametral_pitch(circ_pitch=8);
// diam_pitch3 = diametral_pitch(diam_pitch=16);
function diametral_pitch(circ_pitch, mod, pitch, diam_pitch) =
let( circ_pitch = circular_pitch(pitch, mod, circ_pitch, diam_pitch) )
PI / circ_pitch / INCH;
// Function: module_value()
// Synopsis: Returns tooth density expressed as "module"
// Topics: Gears, Parts
// See Also: spur_gear(), diametral_pitch(), circular_pitch(), module_value()
// Usage:
// mod = module_value(circ_pitch);
// mod = module_value(mod=);
// mod = module_value(diam_pitch=);
// Description:
// Get tooth size expressed as "module". The module is the pitch
// diameter of the gear divided by the number of teeth on the gear. For example, a gear with a pitch
// diameter of 40 mm, with 20 teeth on it will have a modulus of 2 mm. For circ_pitch and mod this
// calculation does not depend on untis. If you give diametral pitch, which is based on inputs, then
// the module is returned in millimeters.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// Example:
// mod1 = module_value(circ_pitch=8);
// mod2 = module_value(mod=2);
// mod3 = module_value(diam_pitch=16);
function module_value(circ_pitch, mod, pitch, diam_pitch) =
let( circ_pitch = circular_pitch(circ_pitch, mod, pitch, diam_pitch) )
circ_pitch / PI;
/// Function: _adendum()
/// Usage:
/// ad = _adendum(circ_pitch, [profile_shift]);
/// ad = _adendum(diam_pitch=, [profile_shift=]);
/// ad = _adendum(mod=, [profile_shift=]);
/// Topics: Gears
/// Description:
/// The height of the top of a gear tooth above the pitch radius circle.
/// Arguments:
/// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
/// profile_shift = Profile shift factor x. Default: 0
/// ---
/// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
/// mod = The module of the gear (pitch diameter / teeth)
/// Example:
/// ad = _adendum(circ_pitch=5);
/// ad = _adendum(mod=2);
/// Example(2D):
/// circ_pitch = 5; teeth = 17;
/// pr = pitch_radius(circ_pitch, teeth);
/// adn = _adendum(circ_pitch=5);
/// #spur_gear2d(circ_pitch=circ_pitch, teeth=teeth);
/// color("black") {
/// stroke(circle(r=pr),width=0.1,closed=true);
/// stroke(circle(r=pr+adn),width=0.1,closed=true);
/// }
function _adendum(
circ_pitch,
profile_shift=0,
shorten=0,
diam_pitch,
mod,
pitch
) =
let( mod = module_value(circ_pitch, mod, pitch, diam_pitch) )
mod * (1 + profile_shift - shorten);
/// Function: _dedendum()
/// Usage:
/// ddn = _dedendum(circ_pitch=, [clearance], [profile_shift]);
/// ddn = _dedendum(diam_pitch=, [clearance=], [profile_shift=]);
/// ddn = _dedendum(mod=, [clearance=], [profile_shift=]);
/// Topics: Gears
/// Description:
/// The depth of the gear tooth valley, below the pitch radius.
/// Arguments:
/// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
/// clearance = If given, sets the clearance between meshing teeth. Default: module/4
/// profile_shift = Profile shift factor x. Default: 0
/// ---
/// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
/// mod = The module of the gear (pitch diameter / teeth)
/// shorten = amount to shorten tip
/// Example:
/// ddn = _dedendum(circ_pitch=5);
/// ddn = _dedendum(mod=2);
/// Example(2D):
/// circ_pitch = 5; teeth = 17;
/// pr = pitch_radius(circ_pitch, teeth);
/// ddn = _dedendum(circ_pitch=5);
/// #spur_gear2d(circ_pitch=circ_pitch, teeth=teeth);
/// color("black") {
/// stroke(circle(r=pr),width=0.1,closed=true);
/// stroke(circle(r=pr-ddn),width=0.1,closed=true);
/// }
function _dedendum(
circ_pitch,
clearance,
profile_shift=0,
diam_pitch,
mod,
pitch
) = let(
mod = module_value(circ_pitch, mod, pitch, diam_pitch),
clearance = default(clearance, 0.25 * mod)
)
mod * (1 - profile_shift) + clearance;
// Function: pitch_radius()
// Synopsis: Returns the pitch radius for a gear.
// Topics: Gears, Parts
// See Also: spur_gear(), diametral_pitch(), circular_pitch(), module_value(), outer_radius()
// Usage:
// pr = pitch_radius(pitch, teeth, [helical]);
// pr = pitch_radius(mod=, teeth=, [helical=]);
// Description:
// Calculates the pitch radius for the gear. Two mated gears will have their centers spaced apart
// by the sum of the two gear's pitch radii.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// teeth = The number of teeth on the gear.
// helical = The helical angle (from vertical) of the teeth on the gear. Default: 0
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// Example:
// pr = pitch_radius(circ_pitch=5, teeth=11);
// pr = pitch_radius(circ_pitch=5, teeth=11, helical=30);
// pr = pitch_radius(diam_pitch=10, teeth=11);
// pr = pitch_radius(mod=2, teeth=20);
// pr = pitch_radius(mod=2, teeth=20, helical=30);
// Example(2D,Med,NoScales,VPT=[-0.20531,0.133721,0.658081],VPR=[0,0,0],VPD=82.6686):
// $fn=144;
// teeth=17; circ_pitch = 5;
// pr = pitch_radius(circ_pitch, teeth);
// stroke(spur_gear2d(circ_pitch, teeth), width=0.2);
// color("blue") dashed_stroke(circle(r=pr), width=0.2);
// color("black") {
// stroke([[0,0],polar_to_xy(pr,45)],
// endcaps="arrow", width=0.3);
// fwd(1)
// text("Pitch Radius", size=1.5,
// halign="center", valign="top");
// }
function pitch_radius(
circ_pitch,
teeth,
helical=0,
mod,
diam_pitch,
pitch
) =
let( circ_pitch = circular_pitch(pitch, mod, circ_pitch, diam_pitch) )
assert(is_finite(helical))
assert(is_finite(circ_pitch))
circ_pitch * teeth / PI / 2 / cos(helical);
// Function: outer_radius()
// Synopsis: Returns the outer radius for a gear.
// Topics: Gears, Parts
// See Also: spur_gear(), diametral_pitch(), circular_pitch(), module_value(), pitch_radius(), outer_radius()
// Usage:
// or = outer_radius(circ_pitch, teeth, [helical=], [clearance=], [internal=], [profile_shift=], [shorten=]);
// or = outer_radius(mod=, teeth=, [helical=], [clearance=], [internal=], [profile_shift=], [shorten=]);
// or = outer_radius(diam_pitch=, teeth=, [helical=], [clearance=], [internal=], [profile_shift=], [shorten=]);
// Description:
// Calculates the outer radius for the gear. The gear fits entirely within a cylinder of this radius, unless
// it has been strongly profile shifted, in which case it will be undersized due to tip clipping.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// teeth = The number of teeth on the gear.
// ---
// clearance = If given, sets the clearance between meshing teeth. Default: module/4
// profile_shift = Profile shift factor x. Default: "auto"
// pressure_angle = Pressure angle. Default: 20
// helical = The helical angle (from vertical) of the teeth on the gear. Default: 0
// shorten = Shortening factor, needed to maintain clearance with profile shifting. Default: 0
// internal = If true, calculate for an internal gear.
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// Example:
// or = outer_radius(circ_pitch=5, teeth=20);
// or = outer_radius(circ_pitch=5, teeth=20, helical=30);
// or = outer_radius(diam_pitch=10, teeth=17);
// or = outer_radius(mod=2, teeth=16);
// Example(2D,Med,NoScales,VPT=[-0.20531,0.133721,0.658081],VPR=[0,0,0],VPD=82.6686):
// $fn=144;
// teeth=17; circ_pitch = 5;
// or = outer_radius(circ_pitch, teeth);
// stroke(spur_gear2d(circ_pitch, teeth), width=0.2);
// color("blue") dashed_stroke(circle(r=or), width=0.2);
// color("black") {
// stroke([[0,0],polar_to_xy(or,45)],
// endcaps="arrow", width=0.3);
// fwd(1)
// text("Outer Radius", size=1.5,
// halign="center", valign="top");
// }
function outer_radius(circ_pitch, teeth, clearance, internal=false, helical=0, profile_shift="auto", pressure_angle=20, shorten=0, mod, pitch, diam_pitch) =
let(
circ_pitch = circular_pitch(pitch, mod, circ_pitch, diam_pitch),
profile_shift = auto_profile_shift(teeth, pressure_angle, helical, profile_shift=profile_shift)
)
pitch_radius(circ_pitch, teeth, helical) + (
internal
? _dedendum(circ_pitch, clearance, profile_shift=-profile_shift)
: _adendum(circ_pitch, profile_shift=profile_shift, shorten=shorten)
);
/// Function: _root_radius()
/// Usage:
/// rr = _root_radius(circ_pitch, teeth, [helical], [clearance=], [internal=], [profile_shift=]);
/// rr = _root_radius(diam_pitch=, teeth=, [helical=], [clearance=], [internal=], [profile_shift=]);
/// rr = _root_radius(mod=, teeth=, [helical=], [clearance=], [internal=], [profile_shift=]);
/// Topics: Gears
/// Description:
/// Calculates the root radius for the gear, at the base of the dedendum. Does not apply auto profile shifting.
/// Arguments:
/// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
/// teeth = The number of teeth on the gear.
/// ---
/// clearance = If given, sets the clearance between meshing teeth. Default: module/4
/// internal = If true, calculate for an internal gear.
/// helical = The helical angle (from vertical) of the teeth on the gear. Default: 0
/// profile_shift = Profile shift factor x. Default:0
/// mod = The module of the gear (pitch diameter / teeth)
/// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
/// Example:
/// rr = _root_radius(circ_pitch=5, teeth=11);
/// rr = _root_radius(circ_pitch=5, teeth=16, helical=30);
/// rr = _root_radius(diam_pitch=10, teeth=11);
/// rr = _root_radius(mod=2, teeth=16);
/// Example(2D):
/// pr = _root_radius(circ_pitch=5, teeth=11);
/// #spur_gear2d(pitch=5, teeth=11);
/// color("black")
/// stroke(circle(r=pr),width=0.1,closed=true);
function _root_radius(circ_pitch, teeth, clearance, internal=false, helical=0, profile_shift=0, diam_pitch, mod, pitch) =
let( circ_pitch = circular_pitch(pitch, mod, circ_pitch, diam_pitch) )
pitch_radius(circ_pitch, teeth, helical) - (
internal
? _adendum(circ_pitch, profile_shift=-profile_shift)
: _dedendum(circ_pitch, clearance, profile_shift=profile_shift)
);
/// Function: _base_radius()
/// Usage:
/// br = _base_radius(circ_pitch, teeth, [pressure_angle], [helical]);
/// br = _base_radius(diam_pitch=, teeth=, [pressure_angle=], [helical=]);
/// br = _base_radius(mod=, teeth=, [pressure_angle=], [helical=]);
/// Topics: Gears
/// Description:
/// Get the base circle for involute teeth, at the base of the teeth.
/// Arguments:
/// pitch = The circular pitch, the distance between teeth centers around the pitch circle.
/// teeth = The number of teeth on the gear.
/// pressure_angle = Pressure angle in degrees. Controls how straight or bulged the tooth sides are.
/// helical = The helical angle (from vertical) of the teeth on the gear. Default: 0
/// ---
/// mod = The module of the gear (pitch diameter / teeth)
/// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
/// Example:
/// br = _base_radius(circ_pitch=5, teeth=20, pressure_angle=20);
/// br = _base_radius(circ_pitch=5, teeth=20, pressure_angle=20, helical=30);
/// br = _base_radius(diam_pitch=10, teeth=20, pressure_angle=20);
/// br = _base_radius(mod=2, teeth=18, pressure_angle=20);
/// Example(2D):
/// pr = _base_radius(circ_pitch=5, teeth=11);
/// #spur_gear2d(circ_pitch=5, teeth=11);
/// color("black")
/// stroke(circle(r=pr),width=0.1,closed=true);
function _base_radius(circ_pitch, teeth, pressure_angle=20, helical=0, diam_pitch, mod, pitch) =
let(
circ_pitch = circular_pitch(pitch, mod, circ_pitch, diam_pitch),
trans_pa = atan(tan(pressure_angle)/cos(helical))
)
pitch_radius(circ_pitch, teeth, helical) * cos(trans_pa);
// Function: bevel_pitch_angle()
// Synopsis: Returns the pitch cone angle for a bevel gear.
// Topics: Gears, Parts
// See Also: bevel_gear(), pitch_radius(), outer_radius()
// Usage:
// ang = bevel_pitch_angle(teeth, mate_teeth, [drive_angle=]);
// Description:
// Returns the correct pitch cone angle for a bevel gear with a given number of teeth, that is
// matched to another bevel gear with a (possibly different) number of teeth.
// Arguments:
// teeth = Number of teeth that this gear has.
// mate_teeth = Number of teeth that the matching gear has.
// drive_angle = Angle between the drive shafts of each gear. Default: 90º.
// Example:
// ang = bevel_pitch_angle(teeth=18, mate_teeth=30);
// Example(2D):
// t1 = 13; t2 = 19; pitch=5;
// pang = bevel_pitch_angle(teeth=t1, mate_teeth=t2, drive_angle=90);
// color("black") {
// zrot_copies([0,pang])
// stroke([[0,0,0], [0,-20,0]],width=0.2);
// stroke(arc(r=3, angle=[270,270+pang]),width=0.2);
// }
// #bevel_gear(
// pitch=5, teeth=t1, mate_teeth=t2,
// spiral=0, cutter_radius=1000,
// slices=12, anchor="apex", orient=BACK
// );
function bevel_pitch_angle(teeth, mate_teeth, drive_angle=90) =
atan(sin(drive_angle)/((mate_teeth/teeth)+cos(drive_angle)));
// Function: worm_gear_thickness()
// Synopsis: Returns the thickness for a worm gear.
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), pitch_radius(), outer_radius()
// Usage:
// thick = worm_gear_thickness(pitch, teeth, worm_diam, [worm_arc=], [crowning=], [clearance=]);
// thick = worm_gear_thickness(mod=, teeth=, worm_diam=, [worm_arc=], [crowning=], [clearance=]);
// Description:
// Calculate the thickness of the worm gear.
// Arguments:
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle. Default: 5
// teeth = Total number of teeth along the rack. Default: 30
// worm_diam = The pitch diameter of the worm gear to match to. Default: 30
// ---
// worm_arc = The arc of the worm to mate with, in degrees. Default: 45 degrees
// pressure_angle = Pressure angle in degrees. Controls how straight or bulged the tooth sides are. Default: 20º
// crowning = The amount to oversize the virtual hobbing cutter used to make the teeth, to add a slight crowning to the teeth to make them fit the work easier. Default: 1
// clearance = Clearance gap at the bottom of the inter-tooth valleys. Default: module/4
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// Example:
// thick = worm_gear_thickness(circ_pitch=5, teeth=36, worm_diam=30);
// thick = worm_gear_thickness(mod=2, teeth=28, worm_diam=25);
// Example(2D):
// circ_pitch = 5;
// teeth = 17;
// worm_diam = 30;
// worm_starts = 2;
// worm_arc = 40;
// y = worm_gear_thickness(
// circ_pitch=circ_pitch,
// teeth=teeth,
// worm_diam=worm_diam,
// worm_arc=worm_arc
// );
// #worm_gear(
// circ_pitch=circ_pitch,
// teeth=teeth,
// worm_diam=worm_diam,
// worm_arc=worm_arc,
// worm_starts=worm_starts,
// orient=BACK
// );
// color("black") {
// ycopies(y) stroke([[-25,0],[25,0]], width=0.5);
// stroke([[-20,-y/2],[-20,y/2]],width=0.5,endcaps="arrow");
// }
function worm_gear_thickness(
circ_pitch,
teeth,
worm_diam,
worm_arc=45,
pressure_angle=20,
crowning=0.1,
clearance,
diam_pitch,
mod,
pitch
) = let(
circ_pitch = circular_pitch(pitch, mod, circ_pitch, diam_pitch),
thickness = worm_gear(
circ_pitch=circ_pitch,
teeth=teeth,
worm_diam=worm_diam,
worm_arc=worm_arc,
crowning=crowning,
pressure_angle=pressure_angle,
clearance=clearance,
get_thickness=true
)
) thickness;
// Function: worm_dist()
// Synopsis: Returns the distance between a worm and a worm gear
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), pitch_radius(), outer_radius()
// Usage:
// dist = worm_dist(mod=|diam_pitch=|circ_pitch=, d, starts, teeth, [profile_shift], [pressure_angle=]);
// Description:
// Calculate the distance between the centers of a worm and its mating worm gear, taking account
// possible profile shifting of the worm gear.
// Arguments:
// d = diameter of worm
// starts = number of starts of worm
// teeth = number of teeth on worm gear
// profile_shift = profile shift of worm gear
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// pressure_angle = The pressure angle of the gear.
// backlash = Add extra space to produce a total of 2*backlash between the two gears.
function worm_dist(d,starts,teeth,mod,profile_shift=0,diam_pitch,circ_pitch,pressure_angle=20,backlash=0) =
let(
mod = module_value(mod=mod,diam_pitch=diam_pitch,circ_pitch=circ_pitch),
lead_angle = asin(mod*starts/d),
pitch_diam = mod*teeth/cos(lead_angle)
)
(d+pitch_diam)/2 + profile_shift*mod
// + backlash * (cos(lead_angle)+cos(90-lead_angle)) / tan(pressure_angle);
// + backlash * cos(45-lead_angle) / tan(pressure_angle);
+ backlash * cos(lead_angle) / tan(pressure_angle);
// Function: gear_dist()
// Synopsis: Returns the distance between two gear centers for spur gears or parallel axis helical gears.
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), pitch_radius(), outer_radius()
// Usage:
// dist = gear_dist(mod=|diam_pitch=|circ_pitch=, teeth1, teeth2, [helical], [profile_shift1], [profile_shift2], [pressure_angle=], [backlash=]);
// Description:
// Calculate the distance between the centers of two spur gears gears or helical gears with parallel axes,
// taking into account profile shifting and helical angle. You can give the helical angle as either positive or negative.
// If you set one of the tooth counts to zero than that gear will be treated as a rack and the distance returned is the
// distance between the rack's pitch line and the gear's center. If you set internal1 or internal2 to true then the
// specified gear is a ring gear; the returned distance is still the distance between the centers of the gears. Note that
// for a regular gear and ring gear to be compatible the ring gear must have more teeth and at least as much profile shift
// as the regular gear.
// .
// The backlash parameter computes the distance offset that produces a total backlash of `2*backlash` in the
// two gear mesh system. This is equivalent to giving the same backlash argument to both gears.
// Arguments:
// teeth1 = Total number of teeth in the first gear. If given 0, we assume this is a rack or worm.
// teeth2 = Total number of teeth in the second gear. If given 0, we assume this is a rack or worm.
// helical = The value of the helical angle (from vertical) of the teeth on the two gears (either sign). Default: 0
// profile_shift1 = Profile shift factor x for the first gear. Default: 0
// profile_shift2 = Profile shift factor x for the second gear. Default: 0
// --
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// internal1 = first gear is an internal (ring) gear. Default: false
// internal2 = second gear is an internal (ring) gear. Default: false
// pressure_angle = The pressure angle of the gear.
// backlash = Add extra space to produce a total of 2*backlash between the two gears.
// Example(2D,NoAxes): Spur gears (with automatic profile shifting on both)
// circ_pitch=5; teeth1=7; teeth2=24;
// d = gear_dist(circ_pitch=circ_pitch, teeth1, teeth2);
// spur_gear2d(circ_pitch, teeth1, gear_spin=-90);
// right(d) spur_gear2d(circ_pitch, teeth2, gear_spin=90-180/teeth2);
// Example(3D,NoAxes,Med,VPT=[23.9049,5.42594,-4.68026],VPR=[64.8,0,353.5],VPD=140): Helical gears (with auto profile shifting on one of the gears)
// circ_pitch=5; teeth1=7; teeth2=24; helical=37;
// d = gear_dist(circ_pitch=circ_pitch, teeth1, teeth2, helical);
// spur_gear(circ_pitch, teeth1, helical=helical, gear_spin=-90,slices=15);
// right(d) spur_gear(circ_pitch, teeth2, helical=-helical, gear_spin=-90-180/teeth2,slices=9);
// Example(2D,NoAxes): Disable Auto Profile Shifting on the smaller gear
// circ_pitch=5; teeth1=7; teeth2=24;
// d = gear_dist(circ_pitch=circ_pitch, teeth1, teeth2, profile_shift1=0);
// spur_gear2d(circ_pitch, teeth1, profile_shift=0, gear_spin=-90);
// right(d) spur_gear2d(circ_pitch, teeth2, gear_spin=90-180/teeth2);
// Example(2D,NoAxes): Manual Profile Shifting
// circ_pitch=5; teeth1=7; teeth2=24; ps1 = 0.5; ps2 = -0.2;
// d = gear_dist(circ_pitch=circ_pitch, teeth1, teeth2, profile_shift1=ps1, profile_shift2=ps2);
// spur_gear2d(circ_pitch, teeth1, profile_shift=ps1, gear_spin=-90);
// right(d) spur_gear2d(circ_pitch, teeth2, profile_shift=ps2, gear_spin=90-180/teeth2);
// Example(2D,NoAxes): Profile shifted gear and a rack
// mod=3; teeth=8;
// d = gear_dist(mod=mod, teeth, 0);
// rack2d(mod=mod, teeth=5, bottom=9);
// back(d) spur_gear2d(mod=mod, teeth=teeth, gear_spin=180/teeth);
// Example(3D,Med,NoAxes,VPT=[-0.0608489,1.3772,-3.68839],VPR=[63.4,0,29.7],VPD=113.336): Profile shifted helical gear and rack
// mod=3; teeth=8; helical=29;
// d = gear_dist(mod=mod, teeth, 0, helical);
// rack(mod=mod, teeth=5, helical=helical, orient=FWD);
// color("lightblue")
// fwd(d) spur_gear(mod=mod, teeth=teeth, helical=-helical, gear_spin=180/teeth);
function gear_dist(
teeth1,
teeth2,
helical=0,
profile_shift1,
profile_shift2,
internal1=false,
internal2=false,
backlash = 0,
pressure_angle=20,
diam_pitch,
circ_pitch,
mod
) =
assert(all_nonnegative([teeth1,teeth2]),"Must give nonnegative values for teeth")
assert(teeth1>0 || teeth2>0, "One of the teeth counts must be nonzero")
assert(is_bool(internal1))
assert(is_bool(internal2))
assert(is_finite(helical))
assert(!(internal1&&internal2), "Cannot specify both gears as internal")
assert(!(internal1 || internal2) || (teeth1>0 && teeth2>0), "Cannot specify internal gear with rack (zero tooth count)")
let(
mod = module_value(mod=mod,circ_pitch= circ_pitch, diam_pitch=diam_pitch),
profile_shift1 = auto_profile_shift(teeth1,pressure_angle,helical,profile_shift=profile_shift1),
profile_shift2 = auto_profile_shift(teeth2,pressure_angle,helical,profile_shift=profile_shift2),
teeth1 = internal2? -teeth1 : teeth1,
teeth2 = internal1? -teeth2 : teeth2
)
assert(teeth1+teeth2>0, "Internal gear must have more teeth than the mated external gear")
let(
profile_shift1 = internal2? -profile_shift1 : profile_shift1,
profile_shift2 = internal1? -profile_shift2 : profile_shift2
)
assert(!(internal1||internal2) || profile_shift1+profile_shift2>=0, "Internal gear must have profile shift equal or greater than mated external gear")
teeth1==0 || teeth2==0? pitch_radius(mod=mod, teeth=teeth1+teeth2, helical=helical) + (profile_shift1+profile_shift2)*mod
:
let(
pa_eff = _working_pressure_angle(teeth1,profile_shift1,teeth2,profile_shift2,pressure_angle,helical),
pa_transv = atan(tan(pressure_angle)/cos(helical))
)
mod*(teeth1+teeth2)*cos(pa_transv)/cos(pa_eff)/cos(helical)/2
+ (internal1||internal2?-1:1) * backlash*cos(helical)/tan(pressure_angle);
function _invol(a) = tan(a) - a*PI/180;
function _working_pressure_angle(teeth1,profile_shift1, teeth2, profile_shift2, pressure_angle, helical) =
let(
pressure_angle = atan(tan(pressure_angle)/cos(helical))
)
teeth1==0 || teeth2==0 ? pressure_angle
:
let(
rhs = 2*(profile_shift1+profile_shift2)/(teeth1+teeth2)*cos(helical)*tan(pressure_angle) + _invol(pressure_angle)
)
assert(rhs>0, "Total profile shift is too small, so working pressure angle is negative, and no valid gear separation exists")
let(
pa_eff = root_find(function (x) _invol(x)-rhs, 1, 75)
)
pa_eff;
// Function: gear_dist_skew()
// Usage:
// Synopsis: Returns the distance between two helical gear centers with skew axes.
// Topics: Gears, Parts
// See Also: gear_dist(), worm(), worm_gear(), pitch_radius(), outer_radius()
// Usage:
// dist = gear_dist_skew(mod=|diam_pitch=|circ_pitch=, teeth1, teeth2, helical1, helical2, [profile_shift1], [profile_shift2], [pressure_angle=]
// Description:
// Calculate the distance between two helical gears that mesh with non-parallel axes, taking into account
// profile shift and the helical angles.
// Arguments:
// teeth1 = Total number of teeth in the first gear. If given 0, we assume this is a rack or worm.
// teeth2 = Total number of teeth in the second gear. If given 0, we assume this is a rack or worm.
// helical1 = The helical angle (from vertical) of the teeth on the first gear.
// helical1 = The helical angle (from vertical) of the teeth on the second gear.
// profile_shift1 = Profile shift factor x for the first gear. Default: "auto"
// profile_shift2 = Profile shift factor x for the second gear. Default: "auto"
// --
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// mod = The module of the gear (pitch diameter / teeth)
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// pressure_angle = The pressure angle of the gear.
// backlash = Add extra space to produce a total of 2*backlash between the two gears.
// Example(3D,Med,NoAxes,VPT=[-0.302111,3.7924,-9.252],VPR=[55,0,25],VPD=155.556): Non-parallel Helical Gears (without any profile shifting)
// circ_pitch=5; teeth1=15; teeth2=24; ha1=45; ha2=30; thick=10;
// d = gear_dist_skew(circ_pitch=circ_pitch, teeth1, teeth2, helical1=ha1, helical2=ha2);
// left(d/2) spur_gear(circ_pitch, teeth1, helical=ha1, thickness=thick, gear_spin=-90);
// right(d/2) xrot(ha1+ha2) spur_gear(circ_pitch, teeth2, helical=ha2, thickness=thick, gear_spin=90-180/teeth2);
function gear_dist_skew(teeth1,teeth2,helical1,helical2,profile_shift1,profile_shift2,pressure_angle=20,
mod, circ_pitch, diam_pitch, backlash=0) =
assert(all_nonnegative([teeth1,teeth2]),"Must give nonnegative values for teeth")
assert(teeth1>0 || teeth2>0, "One of the teeth counts must be nonzero")
let(
profile_shift1 = auto_profile_shift(teeth1,pressure_angle,helical1,profile_shift=profile_shift1),
profile_shift2 = auto_profile_shift(teeth2,pressure_angle,helical2,profile_shift=profile_shift2),
mod = module_value(circ_pitch=circ_pitch, diam_pitch=diam_pitch, mod=mod)
)
teeth1==0 || teeth2==0? pitch_radius(mod=mod, teeth=teeth1+teeth2, helical=teeth1?helical1:helical2) + (profile_shift1+profile_shift2)*mod
:
let(
pa_normal_eff = _working_normal_pressure_angle_skew(teeth1,profile_shift1,helical1,teeth2,profile_shift2,helical2,pressure_angle),
dist_adj = 0.5*(teeth1/cos(helical1)^3+teeth2/cos(helical2)^3)*(cos(pressure_angle)/cos(pa_normal_eff)-1)
)
mod*(teeth1/2/cos(helical1)+teeth2/2/cos(helical2)+dist_adj)
// This expression is a guess based on finding the cross section where pressure angles match so that there is a single
// pressure angle to reference the movement by.
+ backlash * cos((helical1-helical2)/2) / tan(pressure_angle);
function _working_normal_pressure_angle_skew(teeth1,profile_shift1,helical1, teeth2, profile_shift2, helical2, pressure_angle) =
let(
inv = function(a) tan(a) + a*PI/180,
rhs = 2*(profile_shift1+profile_shift2)/(teeth1/cos(helical1)^3+teeth2/cos(helical2)^3)*tan(pressure_angle) + _invol(pressure_angle),
pa_eff_normal = root_find(function (x) _invol(x)-rhs, 5, 75)
)
pa_eff_normal;
// Function: gear_skew_angle()
// Usage:
// ang = gear_skew_angle(teeth1, teeth2, helical1, helical2, [profile_shift1], [profile_shift2], [pressure_angle=]
// Synopsis: Returns corrected skew angle between two profile shifted helical gears.
// Description:
// Compute the correct skew angle between the axes of two profile shifted helical gears. When profile shifting is zero, or when one of
// the gears is a rack, this angle is simply the sum of the helical angles of the two gears. But with profile shifted gears, a small
// correction to the skew angle is needed for proper meshing.
// Arguments:
// teeth1 = Total number of teeth in the first gear. If given 0, we assume this is a rack or worm.
// teeth2 = Total number of teeth in the second gear. If given 0, we assume this is a rack or worm.
// helical1 = The helical angle (from vertical) of the teeth on the first gear.
// helical1 = The helical angle (from vertical) of the teeth on the second gear.
// profile_shift1 = Profile shift factor x for the first gear. Default: "auto"
// profile_shift2 = Profile shift factor x for the second gear. Default: "auto"
// --
// pressure_angle = The pressure angle of the gear.
// Example(3D,Med,NoAxes,VPT=[-2.62091,2.01048,-1.31405],VPR=[55,0,25],VPD=74.4017): These gears are auto profile shifted and as a result, do not mesh at the sum of their helical angles, but at 2.5 degrees more.
// circ_pitch=5; teeth1=12; teeth2=7; ha1=25; ha2=30; thick=10;
// d = gear_dist_skew(circ_pitch=circ_pitch, teeth1, teeth2, ha1, ha2);
// ang = gear_skew_angle(teeth1, teeth2, helical1=ha1, helical2=ha2); // Returns 57.7
// left(d/2)
// spur_gear(circ_pitch, teeth1, helical=ha1, thickness=thick, gear_spin=-90);
// right(d/2) color("lightblue")
// xrot(ang) spur_gear(circ_pitch, teeth2, helical=ha2, thickness=thick, gear_spin=90-180/teeth2);
function gear_skew_angle(teeth1,teeth2,helical1,helical2,profile_shift1,profile_shift2,pressure_angle=20) =
assert(all_nonnegative([teeth1,teeth2]),"Must give nonnegative values for teeth")
assert(teeth1>0 || teeth2>0, "One of the teeth counts must be nonzero")
let(
mod = 1, // This is independent of module size
profile_shift1 = auto_profile_shift(teeth1,pressure_angle,helical1,profile_shift=profile_shift1),
profile_shift2 = auto_profile_shift(teeth2,pressure_angle,helical2,profile_shift=profile_shift2)
)
profile_shift1==0 && profile_shift2==0 ? helical1+helical2
: teeth1==0 || teeth2==0 ? helical1+helical2
: let(
a = gear_dist_skew(mod=mod,teeth1,teeth2,helical1,helical2,profile_shift1,profile_shift2,pressure_angle=pressure_angle),
b = gear_dist_skew(mod=mod,teeth1,teeth2,helical1,helical2,0,0,pressure_angle=pressure_angle),
d1 = 2*pitch_radius(mod=mod,teeth=teeth1,helical=helical1),
d2 = 2*pitch_radius(mod=mod,teeth=teeth2,helical=helical2),
dw1 = 2*a*d1/(d1+d2),
dw2 = 2*a*d2/(d1+d2),
beta1 = atan(dw1/d1*tan(helical1)),
beta2 = atan(dw2/d2*tan(helical2))
)
beta1+beta2;
// Function: get_profile_shift()
// Usage:
// total_shift = get_profile_shift(mod=|diam_pitch=|circ_pitch=, desired, teeth1, teeth2, [helical], [pressure_angle=],
// Synopsis: Returns total profile shift needed to achieve a desired spacing between two gears
// Description:
// Compute the total profile shift, split between two gears, needed to place those gears with a specified separation.
// If the requested separation is too small, returns NaN. Note that the profile shift returned may also be impractically
// large or small and does not necessarily lead to a valid gear configuration. You will need to split the profile shift
// between the two gears. Note that for helical gears, much more adjustment is available by modifying the helical angle.
// Arguments:
// desired = desired gear center separation
// teeth1 = number of teeth on first gear
// teeth2 = number of teeth on second gear
// helical = The helical angle (from vertical) of the teeth on the gear. Default: 0
// ---
// mod = The module of the gear (pitch diameter / teeth)
// diam_pitch = The diametral pitch, or number of teeth per inch of pitch diameter. Note that the diametral pitch is a completely different thing than the pitch diameter.
// circ_pitch = The circular pitch, the distance between teeth centers around the pitch circle.
// pressure_angle = normal pressure angle of gear teeth. Default: 20
// Example(2D,Med,NoAxes,VPT=[37.0558,0.626722,9.78411],VPR=[0,0,0],VPD=496): For a pair of module 4 gears with 19, and 37 teeth, the separation without profile shifting is 112. Suppose we want it instead to be 115. A positive profile shift, split evenly between the gears, achieves the goal, as shown by the red rectangle, with width 115.
// teeth1=37;
// teeth2=19;
// mod=4;
// desired=115;
// pshift = get_profile_shift(desired,teeth1,teeth2,mod=mod); // Returns 0.82
// ps1 = pshift/2;
// ps2 = pshift/2;
// shorten=gear_shorten(teeth1,teeth2,0,ps1,ps2); // Returns 0.07
// d = gear_dist(mod=mod, teeth1,teeth2,0,ps1,ps2);
// spur_gear2d(mod=mod,teeth=teeth1,profile_shift=ps1,shorten=shorten,gear_spin=-90,shaft_diam=5);
// right(d)
// spur_gear2d(mod=mod,teeth=teeth2,profile_shift=ps2,shorten=shorten,gear_spin=-90,shaft_diam=5);
// stroke([rect([desired,40], anchor=LEFT)],color="red");
// Example(2D,Med,NoAxes,VPT=[37.0558,0.626722,9.78411],VPR=[0,0,0],VPD=496): For the same pair of module 4 gears with 19, and 37 teeth, suppose we want a closer spacing of 110 instead of 112. A positive profile shift does the job, as shown by the red rectangle with width 110. More of the negative shift is assigned to the large gear, to avoid undercutting the smaller gear.
// teeth1=37;
// teeth2=19;
// mod=4;
// desired=110;
// pshift = get_profile_shift(desired,teeth1,teeth2,mod=mod); // Returns -0.46
// ps1 = 0.8*pshift;
// ps2 = 0.2*pshift;
// shorten=gear_shorten(teeth1,teeth2,0,ps1,ps2); // Returns 0.04
// d = gear_dist(mod=mod, teeth1,teeth2,0,ps1,ps2);
// spur_gear2d(mod=mod,teeth=teeth1,profile_shift=ps1,shorten=shorten,gear_spin=-90,shaft_diam=5);
// right(d)
// spur_gear2d(mod=mod,teeth=teeth2,profile_shift=ps2,shorten=shorten,gear_spin=-90,shaft_diam=5);
// stroke([rect([desired,40], anchor=LEFT)],color="red");
function get_profile_shift(desired,teeth1,teeth2,helical=0,pressure_angle=20,mod,diam_pitch,circ_pitch) =
let(
mod = module_value(mod=mod, circ_pitch=circ_pitch, diam_pitch=diam_pitch),
teethsum = teeth1+teeth2,
pressure_angle_trans = atan(tan(pressure_angle)/cos(helical)),
y = desired/mod - teethsum/2/cos(helical),
thing=teethsum*cos(pressure_angle_trans) / (teethsum+2*y*cos(helical)),
pa_eff = acos(teethsum*cos(pressure_angle_trans) / (teethsum+2*y*cos(helical)))
)
teethsum * (_invol(pa_eff)-_invol(pressure_angle_trans))/2/tan(pressure_angle);
// Function: auto_profile_shift()
// Synopsis: Returns the recommended profile shift for a gear.
// Topics: Gears, Parts
// See Also: worm(), worm_gear(), pitch_radius(), outer_radius()
// Usage:
// x = auto_profile_shift(teeth, [pressure_angle], [helical], [profile_shift=]);
// x = auto_profile_shift(teeth, [pressure_angle], [helical], get_min=);
// x = auto_profile_shift(teeth, min_teeth=);
// Description:
// Calculates the recommended profile shift to avoid gear tooth undercutting. You can set `min_teeth` to a
// value to allow small undercutting, and only activate the profile shift for more extreme cases. Is is common
// practice to make gears with 15-17 teeth with undercutting with the standard 20 deg pressure angle.
// .
// The `get_min` argument returns the minimum profile shift needed to avoid undercutting for the specified
// number of teeth. This will be a negative value for gears with a large number of teeth; such gears can
// be given a negative profile shift without undercutting.
// Arguments:
// teeth = Total number of teeth in the gear.
// pressure_angle = The pressure angle of the gear.
// helical = helical angle
// ---
// min_teeth = If given, the minimum number of teeth on a gear that has acceptable undercut.
// get_min = If true then return the minimum profile shift to avoid undercutting, which may be a negative value for large gears.
// profile_shift = If numerical then just return this value; if "auto" or not given then compute the automatic profile shift.
function auto_profile_shift(teeth, pressure_angle=20, helical=0, min_teeth, profile_shift, get_min=false) =
assert(is_undef(profile_shift) || is_finite(profile_shift) || profile_shift=="auto", "Profile shift must be \"auto\" or a number")
is_num(profile_shift) ? profile_shift
: teeth==0 ? 0
: let(
pressure_angle=atan(tan(pressure_angle)/cos(helical)),
min_teeth = default(min_teeth, 2 / sin(pressure_angle)^2)
)
!get_min && teeth > floor(min_teeth)? 0
: (1 - (teeth / min_teeth))/cos(helical);
// Function: gear_shorten()
// Usage:
// shorten = gear_shorten(teeth1, teeth2, [helical], [profile_shift1], [profile_shift2], [pressure_angle=]);
// Synopsis: Returns the tip shortening parameter for profile shifted parallel axis gears.
// Description:
// Compute the gear tip shortening factor for gears that have profile shifts. This factor depends on both
// gears in a pair and when applied, will results in teeth that meet the specified clearance distance.
// Generally if you don't apply it the teeth clearance will be decreased due to the profile shifting.
// Because it operates pairwise, if a gear mates with more than one other gear, you may have to decide
// which shortening factor to use. The shortening factor is independent of the size of the teeth.
// Arguments:
// teeth1 = number of teeth on first gear
// teeth2 = number of teeth on second gear
// helical = The helical angle (from vertical) of the teeth on the gear. Default: 0
// profile_shift1 = Profile shift factor x for the first gear. Default: "auto"
// profile_shift2 = Profile shift factor x for the second gear. Default: "auto"
// ---
// pressure_angle = normal pressure angle of gear teeth. Default: 20
// Example(2D,Med,VPT=[53.9088,1.83058,26.0319],VPR=[0,0,0],VPD=140): Big profile shift eliminates the clearance between the teeth
// teeth1=25;
// teeth2=19;
// mod=4;
// ps1 = 0.75;
// ps2 = 0.75;
// d = gear_dist(mod=mod, teeth1,teeth2,0,ps1,ps2);
// color("lightblue")
// spur_gear2d(mod=mod,teeth=teeth1,profile_shift=ps1,gear_spin=-90);
// right(d)
// spur_gear2d(mod=mod,teeth=teeth2,profile_shift=ps2,gear_spin=-90);
// Example(2D,Med,VPT=[53.9088,1.83058,26.0319],VPR=[0,0,0],VPD=140,NoAxes): Applying the correct shortening factor restores the clearance to its normal value.
// teeth1=25;
// teeth2=19;
// mod=4;
// ps1 = 0.75;
// ps2 = 0.75;
// d = gear_dist(mod=mod, teeth1,teeth2,0,ps1,ps2);
// shorten=gear_shorten(teeth1,teeth2,0,ps1,ps2);
// color("lightblue")
// spur_gear2d(mod=mod,teeth=teeth1,profile_shift=ps1,shorten=shorten,gear_spin=-90);
// right(d)
// spur_gear2d(mod=mod,teeth=teeth2,profile_shift=ps2,shorten=shorten,gear_spin=-90);
function gear_shorten(teeth1,teeth2,helical=0,profile_shift1="auto",profile_shift2="auto",pressure_angle=20) =
teeth1==0 || teeth2==0 ? 0
: let(
profile_shift1 = auto_profile_shift(teeth1,pressure_angle,helical,profile_shift=profile_shift1),
profile_shift2 = auto_profile_shift(teeth2,pressure_angle,helical,profile_shift=profile_shift2),
ax = gear_dist(mod=1,teeth1,teeth2,helical,profile_shift1,profile_shift2,pressure_angle=pressure_angle),
y = ax - (teeth1+teeth2)/2/cos(helical)
)
profile_shift1+profile_shift2-y;
// Function: gear_shorten_skew()
// Usage:
// shorten = gear_shorten_skew(teeth1, teeth2, helical1, helical2, [profile_shift1], [profile_shift2], [pressure_angle=]);
// Synopsis: Returns the tip shortening parameter for profile shifted skew axis helical gears.
// Description:
// Compute the gear tip shortening factor for skew axis helical gears that have profile shifts. This factor depends on both
// gears in a pair and when applied, will results in teeth that meet the specified clearance distance.
// Generally if you don't apply it the teeth clearance will be decreased due to the profile shifting.
// Because it operates pairwise, if a gear mates with more than one other gear, you may have to decide
// which shortening factor to use. The shortening factor is independent of the size of the teeth.
// Arguments:
// teeth1 = Total number of teeth in the first gear. If given 0, we assume this is a rack or worm.
// teeth2 = Total number of teeth in the second gear. If given 0, we assume this is a rack or worm.
// helical1 = The helical angle (from vertical) of the teeth on the first gear.
// helical1 = The helical angle (from vertical) of the teeth on the second gear.
// profile_shift1 = Profile shift factor x for the first gear. Default: "auto"
// profile_shift2 = Profile shift factor x for the second gear. Default: "auto"
// ---
// pressure_angle = The pressure angle of the gear.
function gear_shorten_skew(teeth1,teeth2,helical1,helical2,profile_shift1="auto",profile_shift2="auto",pressure_angle=20) =
let(
profile_shift1 = auto_profile_shift(teeth1,pressure_angle,helical1,profile_shift=profile_shift1),
profile_shift2 = auto_profile_shift(teeth2,pressure_angle,helical2,profile_shift=profile_shift2),
ax = gear_dist(mod=1,teeth1,teeth2,helical,profile_shift1,profile_shift2,pressure_angle=pressure_angle),
y = ax - (teeth1+teeth2)/2/cos(helical)
)
profile_shift1+profile_shift2-y;
module _show_gear_tooth_profile(
circ_pitch,
teeth,
pressure_angle=20,
profile_shift,
helical=0,
internal=false,
clearance,
backlash=0,
show_verts=false,
diam_pitch,
mod
) {
mod = module_value(circ_pitch=circ_pitch, diam_pitch=diam_pitch, mod=mod);
profile_shift = default(profile_shift, auto_profile_shift(teeth, pressure_angle, helical));
or = outer_radius(mod=mod, teeth=teeth, clearance=clearance, helical=helical, profile_shift=profile_shift, internal=internal);
pr = pitch_radius(mod=mod, teeth=teeth, helical=helical);
rr = _root_radius(mod=mod, teeth=teeth, helical=helical, profile_shift=profile_shift, clearance=clearance, internal=internal);
br = _base_radius(mod=mod, teeth=teeth, helical=helical, pressure_angle=pressure_angle);
tang = 360/teeth;
rang = tang * 1.075;
tsize = (or-rr) / 20;
clear = (1-profile_shift)*mod;
tooth = _gear_tooth_profile(
mod=mod, teeth=teeth,
pressure_angle=pressure_angle,
clearance=clearance,
backlash=backlash,
helical=helical,
internal=internal,
profile_shift=profile_shift
);
$fn=360;
union() {
color("cyan") { // Pitch circle
stroke(arc(r=pr,start=90-rang/2,angle=rang), width=0.05);
zrot(-tang/2*1.10) back(pr) text("pitch", size=tsize, halign="left", valign="center");
}
color("lightgreen") { // Outer and Root circles
stroke(arc(r=or,start=90-rang/2,angle=rang), width=0.05);
stroke(arc(r=rr,start=90-rang/2,angle=rang), width=0.05);
zrot(-tang/2*1.10) back(or) text("tip", size=tsize, halign="left", valign="center");
zrot(-tang/2*1.10) back(rr) text("root", size=tsize, halign="left", valign="center");
}
color("#fcf") { // Base circle
stroke(arc(r=br,start=90-rang/2,angle=rang), width=0.05);
zrot(tang/2*1.10) back(br) text("base", size=tsize, halign="right", valign="center");
}
color("#ddd") { // Clearance area
if (internal) {
dashed_stroke(arc(r=pr+clear, start=90-rang/2, angle=rang), width=0.05);
back((pr+clear+or)/2) text("clearance", size=tsize, halign="center", valign="center");
} else {
dashed_stroke(arc(r=pr-clear, start=90-rang/2, angle=rang), width=0.05);
back((pr-clear+rr)/2) text("clearance", size=tsize, halign="center", valign="center");
}
}
color("#ddd") { // Tooth width markers
stroke([polar_to_xy(min(rr,br)-mod/10,90-180/teeth),polar_to_xy(or+mod/10,90-180/teeth)], width=0.05, closed=true);
stroke([polar_to_xy(min(rr,br)-mod/10,90+180/teeth),polar_to_xy(or+mod/10,90+180/teeth)], width=0.05, closed=true);
}
zrot_copies([0]) { // Tooth profile overlay
stroke(tooth, width=0.1, dots=(show_verts?"dot":false), endcap_color1="green", endcap_color2="red");
}
}
}
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap