MrDraw/SkeinPyPy_NewUI/fabmetheus_utilities/geometry/creation/gear.py
Daid a414a80837 Adding SkeinPyPy_NewUI as development for a brand new user interface.
Experimental, doesn't slice yet, loads of work ahead.
2012-02-20 00:30:49 +01:00

1152 lines
52 KiB
Python

"""
This page is in the table of contents.
The gear script can generate a spur gear couple, a bevel gear couple, a ring gear couple and a rack & pinion couple.
A helix pattern can be added to each gear type. All the gear types have a clearance and all the teeth can be beveled. A keyway, shaft and lightening holes can be added to all the round gears, and rack holes can be added to the rack. The script can output solid gears or only the gear profiles. Both gears of the couple can be generated or just one.
The couple has a pinion gear and a complement.
==Examples==
The link text includes the distinguishing parameters. Each svg page was generated from an xml page of the same root name using carve. For example, gear.svg was generated by clicking 'Carve' on the carve tool panel and choosing gear.xml in the file chooser.
Each generated svg file has the xml fabmetheus element without comments towards the end of the file. To see it, open the svg file in a text editor and search for 'fabmetheus' If you copy that into a new text document, add the line '<?xml version='1.0' ?>' at the beginning and then give it a file name with the extension '.xml', you could then generate another svg file using carve.
===Bevel===
Bevel gear couple.
<a href='../models/xml_models/creation/gear/bevel.svg'>gear operatingAngle=90</a>
===Collar===
Spur gear couple and each gear has a collar.
<a href='../models/xml_models/creation/gear/collar.svg'>gear complementCollarLengthOverFaceWidth='1' pinionCollarLengthOverFaceWidth='1' shaftRadius='5'</a>
===Gear===
Default spur gear with no parameters.
<a href='../models/xml_models/creation/gear/gear.svg'>gear</a>
===Keyway===
Spur gear couple and each gear has a collar and defined keyway.
<a href='../models/xml_models/creation/gear/keyway.svg'>gear complementCollarLengthOverFaceWidth='1' keywayRadius='2' pinionCollarLengthOverFaceWidth='1' shaftRadius='5'</a>
===Rack===
Rack and pinion couple.
<a href='../models/xml_models/creation/gear/rack.svg'>gear teethComplement='0'</a>
===Rack Hole===
Rack and pinion couple, with holes in the rack.
<a href='../models/xml_models/creation/gear/rack_hole.svg'>gear rackHoleRadiusOverWidth='0.2' rackWidthOverFaceWidth='2' teethComplement='0'</a>
===Ring===
Pinion and ring gear.
<a href='../models/xml_models/creation/gear/ring.svg'>gear teethComplement='-23'</a>
===Shaft===
Spur gear couple and each gear has a square shaft hole.
<a href='../models/xml_models/creation/gear/shaft.svg'>gear shaftRadius='5'</a>
===Shaft Top===
Spur gear couple and each gear has a round shaft hole, truncated on top.
<a href='../models/xml_models/creation/gear/shaft_top.svg'>gear shaftRadius='5' shaftSides='13' shaftDepthTop='2'</a>
===Spur Helix===
Spur gear couple with the gear teeth following a helix path.
<a href='../models/xml_models/creation/gear/spur_helix.svg'>gear helixAngle='45'</a>
===Spur Herringbone===
Spur gear couple with the gear teeth following a herringbone path.
<a href='../models/xml_models/creation/gear/spur_herringbone.svg'>gear helixAngle='45' helixType='herringbone'</a>
===Spur Parabolic===
Spur gear couple with the gear teeth following a parabolic path.
<a href='../models/xml_models/creation/gear/spur_parabolic.svg'>gear helixAngle='45' helixType='parabolic'</a>
===Spur Profile===
Spur gear couple profile. Since this is just a horizontal path, it can not be sliced, so the path is then extruded to create a solid which can be sliced and viewed.
<a href='../models/xml_models/creation/gear/spur_profile.svg'>gear id='spurProfile' faceWidth='0' | extrude target='=document.getElementByID(spurProfile)</a>
==Parameters==
===Center Distance===
Default is such that the pitch radius works out to twenty.
Defines the distance between the gear centers.
===Clearance Couplet===
====Clearance Over Wavelength====
Default is 0.1.
Defines the ratio of the clearance over the wavelength of the gear profile. The wavelength is the arc distance between the gear teeth.
====Clearance====
Default is the 'Clearance Over Wavelength' times the wavelength.
Defines the clearance between the gear tooth and the other gear of the couple. If the clearance is zero, the outside of the gear tooth will touch the other gear. If the clearance is too high, the gear teeth will be long and weak.
===Collar Addendum Couplet===
====Collar Addendum Over Radius====
Default is one.
Defines the ratio of the collar addendum over the shaft radius.
====Collar Addendum====
Default is the 'Collar Addendum Over Radius' times the shaft radius.
Defines the collar addendum.
===Complement Collar Length Couplet===
====Complement Collar Length Over Face Width====
Default is zero.
Defines the ratio of the complement collar length over the face width.
====Complement Collar Length====
Default is the 'Complement Collar Length Over Face Width' times the face width.
Defines the complement collar length. If the complement collar length is zero, there will not be a collar on the complement gear.
===Creation Type===
Default is 'both'.
====Both====
When selected, the pinion and complement will be generated.
====Complement====
When selected, only the complement gear or rack will be generated.
====Pinion====
When selected, only the pinion will be generated.
===Face Width===
Default is ten.
Defines the face width.
===Gear Hole Paths===
Default is empty.
Defines the centers of the gear holes. If the gear hole paths parameter is the default empty, then the centers of the gear holes will be generated from other parameters.
===Helix Angle===
Default is zero.
===Helix Path===
Default is empty.
Defines the helix path of the gear teeth. If the helix path is the default empty, then the helix will be generated from the helix angle and helix type.
===Helix Type===
Default is 'basic'.
====Basic====
When selected, the helix will be basic.
====Herringbone====
When selected, the helix will have a herringbone pattern.
====Parabolic====
When selected, the helix will have a parabolic pattern.
===Keyway Radius Couplet===
====Keyway Radius Over Radius====
Default is half.
Defines the ratio of the keyway radius over the shaft radius.
====Keyway Radius====
Default is the 'Keyway Radius Over Radius' times the shaft radius.
Defines the keyway radius. If the keyway radius is zero, there will not be a keyway on the collar.
===Lightening Hole Margin Couplet===
====Lightening Hole Margin Over Rim Dedendum====
Default is one.
Defines the ratio of the lightening hole margin over the rim dedendum.
====Lightening Hole Margin====
Default is the 'Lightening Hole Margin Over Rim Dedendum' times the rim dedendum.
Defines the minimum margin between lightening holes.
===Lightening Hole Minimum Radius===
Default is one.
Defines the minimum radius of the lightening holes.
===Move Type===
Default is 'separate'.
====None====
When selected, the gears will be not be moved and will therefore overlap. Afterwards the write plugin could be used to write each gear to a different file, so they can be fabricated in separate operations.
====Mesh====
When selected, the gears will be separated horizontally so that they just mesh. This is useful to test if the gears mesh properly.
====Separate====
When selected, the gears will be separated horizontally with a gap between them.
====Vertical====
When selected, the gears will be separated vertically.
===Operating Angle===
Default is 180 degrees.
Defines the operating angle between the gear axes. If the operating angle is not 180 degrees, a bevel gear couple will be generated.
===Pinion Collar Length Couplet===
====Pinion Collar Length Over Face Width====
Default is zero.
Defines the ratio of the pinion collar length over the face width.
====Pinion Collar Length====
Default is the 'Pinion Collar Length Over Face Width' times the face width.
Defines the pinion collar length. If the pinion collar length is zero, there will not be a collar on the pinion gear.
===Pitch Radius===
Default is twenty if the pitch radius has not been set. If the center distance is set, the default pitch radius is the center distance times the number of pinion teeth divided by the total number of gear teeth.
Defines the pinion pitch radius.
===Plate Clearance Couplet===
====Plate Clearance Over Length====
Default is 0.2.
Defines the ratio of the plate clearance over the plate length.
====Plate Clearance====
Default is the 'Plate Clearance Over Length' times the plate length.
Defines the clearance between the pinion and the plate of the ring gear. If the clearance is zero, they will touch.
===Plate Length Couplet===
====Plate Length Over Face Width====
Default is half.
Defines the ratio of the plate length over the face width.
====Plate Length====
Default is the 'Plate Length Over Face Width' times the face width.
Defines the length of the plate of the ring gear.
===Pressure Angle===
Default is twenty degrees.
Defines the pressure angle of the gear couple.
===Profile Surfaces===
Default is eleven.
Defines the number of profile surfaces.
===Rack Hole Below Over Width Couplet===
====Rack Hole Below Over Width====
Default is 0.6.
Defines the ratio of the distance below the pitch of the rack holes over the rack width.
====Rack Hole Below====
Default is the 'Rack Hole Below Over Width' times the rack width.
Defines the the distance below the pitch of the rack holes.
===Rack Hole Radius Couplet===
====Rack Hole Radius Over Width====
Default is zero.
Defines the ratio of the rack hole radius over the rack width.
====Rack Hole Radius====
Default is the 'Rack Hole Radius Over Width' times the rack width.
Defines the radius of the rack holes. If the rack hole radius is zero, there won't be any rack holes.
===Rack Hole Step Over Width Couplet===
====Rack Hole Step Over Width====
Default is one.
Defines the ratio of the rack hole step over the rack width.
====Rack Hole Step====
Default is the 'Rack Hole Step Over Width' times the rack width.
Defines the horizontal step distance between the rack holes.
===Rack Length Over Radius Couplet===
====Rack Length Over Radius====
Default is two times pi.
Defines the ratio of the rack length over the pitch radius.
====Rack Length====
Default is the 'Rack Length Over Radius' times the pitch radius.
Defines the rack length.
===Rack Width Couplet===
====Rack Width Over Face Width====
Default is one.
Defines the ratio of the rack width over the face width.
====Rack Width====
Default is the 'Rack Width Over Face Width' times the face width.
Defines the rack width.
===Rim Dedendum Couplet===
====Rim Dedendum Over Radius====
Default is 0.2.
Defines the ratio of the rim dedendum over the pitch radius.
====Rim Dedendum====
Default is the 'Rim Dedendum Over Radius' times the pitch radius.
Defines the rim dedendum of the gear.
===Root Bevel Couplet===
====Root Bevel Over Clearance====
Default is half.
Defines the ratio of the root bevel over the clearance.
====Root Bevel====
Default is the 'Root Bevel Over Clearance' times the clearance.
Defines the bevel at the root of the gear tooth.
===Shaft Depth Bottom Couplet===
====Shaft Depth Bottom Over Radius====
Default is zero.
Defines the ratio of the bottom shaft depth over the shaft radius.
====Shaft Depth Bottom====
Default is the 'Shaft Depth Bottom Over Radius' times the shaft radius.
Defines the bottom shaft depth.
===Shaft Depth Top Couplet===
====Shaft Depth Top Over Radius====
Default is zero.
Defines the ratio of the top shaft depth over the shaft radius.
====Shaft Depth Top====
Default is the 'Shaft Depth Top Over Radius' times the shaft radius.
Defines the top shaft depth.
===Shaft Path===
Default is empty.
Defines the path of the shaft hole. If the shaft path is the default empty, then the shaft path will be generated from the shaft depth bottom, shaft depth top, shaft radius and shaft sides.
===Shaft Radius Couplet===
====Shaft Radius Over Pitch Radius====
Default is zero.
Defines the ratio of the shaft radius over the pitch radius.
====Shaft Radius====
Default is the 'Shaft Radius Over Pitch Radius' times the pitch radius.
Defines the shaft radius. If the shaft radius is zero there will not be a shaft hole.
===Shaft Sides===
Default is four.
Defines the number of shaft sides.
===Teeth Pinion===
Default is seven.
Defines the number of teeth in the pinion.
===Teeth Complement===
Default is seventeen.
Defines the number of teeth in the complement of the gear couple. If the number of teeth is positive, the gear couple will be a spur or bevel type. If the number of teeth is zero, the gear couple will be a rack and pinion. If the number of teeth is negative, the gear couple will be a spur and ring.
===Tip Bevel Couplet===
====Tip Bevel Over Clearance====
Default is 0.1.
Defines the ratio of the tip bevel over the clearance.
====Tip Bevel====
Default is the 'Tip Bevel Over Clearance' times the clearance.
Defines the bevel at the tip of the gear tooth.
===Tooth Thickness Multiplier===
Default is 0.99999.
Defines the amount the thickness of the tooth will multiplied. If when the gears are produced, they mesh too tightly, you can reduce the tooth thickness multiplier so that they mesh with reasonable tightness.
"""
from __future__ import absolute_import
#Init has to be imported first because it has code to workaround the python bug where relative imports don't work if the module is imported as a main module.
import __init__
from fabmetheus_utilities.geometry.creation import extrude
from fabmetheus_utilities.geometry.creation import lineation
from fabmetheus_utilities.geometry.creation import shaft
from fabmetheus_utilities.geometry.creation import solid
from fabmetheus_utilities.geometry.creation import teardrop
from fabmetheus_utilities.geometry.geometry_tools import path
from fabmetheus_utilities.geometry.geometry_utilities.evaluate_elements import setting
from fabmetheus_utilities.geometry.geometry_utilities import evaluate
from fabmetheus_utilities.geometry.geometry_utilities import matrix
from fabmetheus_utilities.geometry.solids import triangle_mesh
from fabmetheus_utilities.vector3 import Vector3
from fabmetheus_utilities.vector3index import Vector3Index
from fabmetheus_utilities import euclidean
import math
__author__ = 'Enrique Perez (perez_enrique@yahoo.com)'
__credits__ = 'Art of Illusion <http://www.artofillusion.org/>'
__date__ = '$Date: 2008/02/05 $'
__license__ = 'GNU Affero General Public License http://www.gnu.org/licenses/agpl.html'
def addBevelGear(derivation, extrudeDerivation, pitchRadius, positives, teeth, vector3GearProfile):
"Get extrude output for a cylinder gear."
totalPitchRadius = derivation.pitchRadiusComplement + derivation.pitchRadius
totalTeeth = derivation.teethPinion + derivation.teethComplement
portionDirections = extrude.getSpacedPortionDirections(extrudeDerivation.interpolationDictionary)
loopLists = extrude.getLoopListsByPath(extrudeDerivation, None, vector3GearProfile[0], portionDirections)
firstLoopList = loopLists[0]
gearOverPinion = float(totalTeeth - teeth) / float(teeth)
thirdLayerHeight = 0.33333333333 * setting.getLayerHeight(derivation.elementNode)
pitchRadian = math.atan(math.sin(derivation.operatingRadian) / (gearOverPinion + math.cos(derivation.operatingRadian)))
coneDistance = pitchRadius / math.sin(pitchRadian)
apex = Vector3(0.0, 0.0, math.sqrt(coneDistance * coneDistance - pitchRadius * pitchRadius))
cosPitch = apex.z / coneDistance
sinPitch = math.sin(pitchRadian)
for loop in firstLoopList:
for point in loop:
alongWay = point.z / coneDistance
oneMinusAlongWay = 1.0 - alongWay
pointComplex = point.dropAxis()
pointComplexLength = abs(pointComplex)
deltaRadius = pointComplexLength - pitchRadius
cosDeltaRadius = cosPitch * deltaRadius
sinDeltaRadius = sinPitch * deltaRadius
pointComplex *= (cosDeltaRadius + pitchRadius) / pointComplexLength
point.x = pointComplex.real
point.y = pointComplex.imag
point.z += sinDeltaRadius
point.x *= oneMinusAlongWay
point.y *= oneMinusAlongWay
addBottomLoop(-thirdLayerHeight, firstLoopList)
topLoop = firstLoopList[-1]
topAddition = []
topZ = euclidean.getTopPath(topLoop) + thirdLayerHeight
oldIndex = topLoop[-1].index
for point in topLoop:
oldIndex += 1
topAddition.append(Vector3Index(oldIndex, 0.8 * point.x, 0.8 * point.y, topZ))
firstLoopList.append(topAddition)
translation = Vector3(0.0, 0.0, -euclidean.getBottomByPaths(firstLoopList))
euclidean.translateVector3Paths(firstLoopList, translation)
geometryOutput = triangle_mesh.getPillarsOutput(loopLists)
positives.append(geometryOutput)
def addBottomLoop(deltaZ, loops):
"Add bottom loop to loops."
bottomLoop = loops[0]
bottomAddition = []
bottomZ = euclidean.getBottomByPath(bottomLoop) + deltaZ
for point in bottomLoop:
bottomAddition.append(Vector3Index(len(bottomAddition), point.x, point.y, bottomZ))
loops.insert(0, bottomAddition)
numberOfVertexes = 0
for loop in loops:
for point in loop:
point.index = numberOfVertexes
numberOfVertexes += 1
def addCollarShaft(collarLength, derivation, elementNode, negatives, positives):
'Add collar.'
if collarLength <= 0.0:
addShaft(derivation, negatives, positives)
return
connectionEnd = Vector3(0.0, 0.0, derivation.faceWidth + collarLength)
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = [Vector3(0.0, 0.0, derivation.faceWidth), connectionEnd]
collarDerivation = extrude.ExtrudeDerivation(copyShallow)
addCollarShaftSetDerivation(collarDerivation, collarLength, derivation, elementNode, negatives, positives)
def addCollarShaftSetDerivation(collarDerivation, collarLength, derivation, elementNode, negatives, positives):
'Add collar and shaft.'
collarSides = evaluate.getSidesMinimumThreeBasedOnPrecision(elementNode, derivation.shaftRimRadius)
collarProfile = euclidean.getComplexPolygon(complex(), derivation.shaftRimRadius, collarSides)
vector3CollarProfile = euclidean.getVector3Path(collarProfile)
extrude.addPositives(collarDerivation, [vector3CollarProfile], positives)
addShaft(derivation, negatives, positives)
drillZ = derivation.faceWidth + 0.5 * collarLength
drillEnd = Vector3(0.0, derivation.shaftRimRadius, drillZ)
drillStart = Vector3(0.0, 0.0, drillZ)
teardrop.addNegativesByRadius(elementNode, drillEnd, negatives, derivation.keywayRadius, drillStart)
def addLighteningHoles(derivation, gearHolePaths, negatives, pitchRadius, positives):
"Add lightening holes."
positiveVertexes = matrix.getVertexes(positives)
bottomPath = euclidean.getTopPath(positiveVertexes)
topPath = euclidean.getBottomByPath(positiveVertexes)
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = [Vector3(0.0, 0.0, bottomPath), Vector3(0.0, 0.0, topPath)]
extrudeDerivation = extrude.ExtrudeDerivation(copyShallow)
vector3LighteningHoles = getLighteningHoles(derivation, gearHolePaths, pitchRadius)
extrude.addNegativesPositives(extrudeDerivation, negatives, vector3LighteningHoles, positives)
def addRackHole(derivation, elementNode, vector3RackProfiles, x):
"Add rack hole to vector3RackProfiles."
rackHole = euclidean.getComplexPolygon(complex(x, -derivation.rackHoleBelow), derivation.rackHoleRadius, -13)
vector3RackProfiles.append(euclidean.getVector3Path(rackHole))
def addRackHoles(derivation, elementNode, vector3RackProfiles):
"Add rack holes to vector3RackProfiles."
if len(derivation.gearHolePaths) > 0:
vector3RackProfiles += derivation.gearHolePaths
return
if derivation.rackHoleRadius <= 0.0:
return
addRackHole(derivation, elementNode, vector3RackProfiles, 0.0)
rackHoleMargin = derivation.rackHoleRadius + derivation.rackHoleRadius
rackHoleSteps = int(math.ceil((derivation.rackDemilength - rackHoleMargin) / derivation.rackHoleStep))
for rackHoleIndex in xrange(1, rackHoleSteps):
x = float(rackHoleIndex) * derivation.rackHoleStep
addRackHole(derivation, elementNode, vector3RackProfiles, -x)
addRackHole(derivation, elementNode, vector3RackProfiles, x)
def addShaft(derivation, negatives, positives):
"Add shaft."
if len(derivation.shaftPath) < 3:
return
positiveVertexes = matrix.getVertexes(positives)
bottomPath = euclidean.getTopPath(positiveVertexes)
topPath = euclidean.getBottomByPath(positiveVertexes)
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = [Vector3(0.0, 0.0, bottomPath), Vector3(0.0, 0.0, topPath)]
extrudeDerivation = extrude.ExtrudeDerivation(copyShallow)
extrude.addNegativesPositives(extrudeDerivation, negatives, [derivation.shaftPath], positives)
def getAxialMargin(circleRadius, numberOfSides, polygonRadius):
'Get axial margin.'
return polygonRadius * math.sin(math.pi / float(numberOfSides)) - circleRadius
def getBevelPath(begin, bevel, center, end):
'Get bevel path.'
centerMinusBegin = center - begin
centerMinusBeginLength = abs(centerMinusBegin)
endMinusCenter = end - center
endMinusCenterLength = abs(endMinusCenter)
endMinusCenter /= endMinusCenterLength
maximumExtensionLength = 0.333333333 * endMinusCenterLength
if centerMinusBeginLength <= bevel * 1.5:
extensionLength = min(maximumExtensionLength, centerMinusBeginLength)
return [complex(center.real, center.imag) + extensionLength * endMinusCenter]
centerMinusBegin *= (centerMinusBeginLength - bevel) / centerMinusBeginLength
extensionLength = min(maximumExtensionLength, bevel)
bevelPath = [complex(center.real, center.imag) + extensionLength * endMinusCenter]
bevelPath.append(begin + centerMinusBegin)
return bevelPath
def getGearPaths(derivation, pitchRadius, teeth, toothProfile):
'Get gear paths.'
if teeth < 0:
return getGearProfileAnnulus(derivation, pitchRadius, teeth, toothProfile)
if teeth == 0:
return [getGearProfileRack(derivation, toothProfile)]
return [getGearProfileCylinder(teeth, toothProfile)]
def getGearProfileAnnulus(derivation, pitchRadius, teeth, toothProfile):
'Get gear profile for an annulus gear.'
gearProfileCylinder = getGearProfileCylinder(teeth, toothProfile)
annulusRadius = derivation.dedendum + derivation.rimDedendum - pitchRadius
return [euclidean.getComplexPolygon(complex(), annulusRadius, -teeth, 0.5 * math.pi), gearProfileCylinder]
def getGearProfileCylinder(teeth, toothProfile):
'Get gear profile for a cylinder gear.'
gearProfile = []
toothAngleRadian = 2.0 * math.pi / float(teeth)
totalToothAngle = 0.0
for toothIndex in xrange(abs(teeth)):
for toothPoint in toothProfile:
gearProfile.append(toothPoint * euclidean.getWiddershinsUnitPolar(totalToothAngle))
totalToothAngle += toothAngleRadian
return gearProfile
def getGearProfileRack(derivation, toothProfile):
'Get gear profile for rack.'
derivation.extraRackDemilength = 0.0
for complexPoint in derivation.helixPath:
derivation.extraRackDemilength = max(abs(derivation.helixHeight * complexPoint.imag), derivation.extraRackDemilength)
rackDemilengthPlus = derivation.rackDemilength
if derivation.faceWidth > 0.0:
derivation.extraRackDemilength *= 1.1
rackDemilengthPlus += derivation.extraRackDemilength
teethRack = int(math.ceil(rackDemilengthPlus / derivation.wavelength))
gearProfile = []
for toothIndex in xrange(-teethRack, teethRack + 1):
translateComplex = complex(-toothIndex * derivation.wavelength, 0.0)
translatedPath = euclidean.getTranslatedComplexPath(toothProfile, translateComplex)
gearProfile += translatedPath
gearProfile = euclidean.getHorizontallyBoundedPath(rackDemilengthPlus, -rackDemilengthPlus, gearProfile)
firstPoint = gearProfile[0]
lastPoint = gearProfile[-1]
rackWidth = derivation.rackWidth
minimumRackWidth = 1.1 * derivation.dedendum
if rackWidth < minimumRackWidth:
rackWidth = minimumRackWidth
print('Warning, rackWidth is too small in getGearProfileRack in gear.')
print('RackWidth will be set to a bit more than the dedendum.')
gearProfile += [complex(lastPoint.real, -rackWidth),complex(firstPoint.real, -rackWidth)]
return gearProfile
def getGeometryOutput(derivation, elementNode):
"Get vector3 vertexes from attribute dictionary."
if derivation == None:
derivation = GearDerivation(elementNode)
creationFirst = derivation.creationType.lower()[: 1]
toothProfileComplement = getToothProfile(derivation, derivation.pitchRadiusComplement, derivation.teethComplement)
pinionProfile = getGearProfileCylinder(derivation.teethPinion, derivation.pinionToothProfile)
complementPaths = getGearPaths(
derivation, derivation.pitchRadiusComplement, derivation.teethComplement, toothProfileComplement)
vector3PinionProfile = euclidean.getVector3Path(pinionProfile)
vector3ComplementPaths = euclidean.getVector3Paths(complementPaths)
translation = Vector3()
moveFirst = derivation.moveType.lower()[: 1]
if moveFirst != 'n':
distance = derivation.pitchRadius
if moveFirst == 'm':
distance += derivation.pitchRadiusComplement
else:
distance += abs(derivation.pitchRadiusComplement)
decimalPlaces = 1 - int(math.floor(math.log10(distance)))
distance += derivation.halfWavelength + derivation.halfWavelength
distance = round(1.15 * distance, decimalPlaces)
translation = Vector3(0.0, -distance)
if derivation.faceWidth <=0.0:
return getPathOutput(
creationFirst, derivation, elementNode, translation, vector3ComplementPaths, vector3PinionProfile)
pitchRadius = derivation.pitchRadius
teeth = derivation.teethPinion
twist = derivation.helixHeight / derivation.pitchRadius
extrudeOutputPinion = getOutputCylinder(
derivation.pinionCollarLength, derivation, elementNode, None, pitchRadius, teeth, twist, [vector3PinionProfile])
if creationFirst == 'p':
return extrudeOutputPinion
teeth = derivation.teethComplement
extrudeOutputSecond = None
if teeth == 0:
extrudeOutputSecond = getOutputRack(derivation, elementNode, vector3ComplementPaths[0])
else:
twist = -derivation.helixHeight / derivation.pitchRadiusComplement
extrudeOutputSecond = getOutputCylinder(
derivation.complementCollarLength,
derivation,
elementNode,
derivation.gearHolePaths,
derivation.pitchRadiusComplement,
teeth,
twist,
vector3ComplementPaths)
if creationFirst == 'c':
return extrudeOutputSecond
gearVertexes = matrix.getVertexes(extrudeOutputSecond)
if moveFirst == 'v':
translation = Vector3(0.0, 0.0, euclidean.getTopPath(gearVertexes))
euclidean.translateVector3Path(matrix.getVertexes(extrudeOutputPinion), translation)
else:
euclidean.translateVector3Path(gearVertexes, translation)
return {'group' : {'shapes' : [extrudeOutputPinion, extrudeOutputSecond]}}
def getGeometryOutputByArguments(arguments, elementNode):
"Get vector3 vertexes from attribute dictionary by arguments."
return getGeometryOutput(None, elementNode)
def getHalfwave(pitchRadius, teeth):
'Get tooth halfwave.'
return pitchRadius * math.pi / float(teeth)
def getHelixComplexPath(derivation, elementNode):
'Set gear helix path.'
helixTypeFirstCharacter = derivation.helixType.lower()[: 1]
if helixTypeFirstCharacter == 'b':
return [complex(), complex(1.0, 1.0)]
if helixTypeFirstCharacter == 'h':
return [complex(), complex(0.5, 0.5), complex(1.0, 0.0)]
if helixTypeFirstCharacter == 'p':
helixComplexPath = []
x = 0.0
xStep = setting.getLayerHeight(elementNode) / derivation.faceWidth
justBelowOne = 1.0 - 0.5 * xStep
while x < justBelowOne:
distanceFromCenter = 0.5 - x
parabolicTwist = 0.25 - distanceFromCenter * distanceFromCenter
helixComplexPath.append(complex(x, parabolicTwist))
x += xStep
helixComplexPath.append(complex(1.0, 0.0))
return helixComplexPath
print('Warning, the helix type was not one of (basic, herringbone or parabolic) in getHelixComplexPath in gear for:')
print(derivation.helixType)
print(derivation.elementNode)
def getLiftedOutput(derivation, geometryOutput):
"Get extrude output for a rack."
if derivation.moveType.lower()[: 1] == 'm':
return geometryOutput
geometryOutputVertexes = matrix.getVertexes(geometryOutput)
translation = Vector3(0.0, 0.0, -euclidean.getBottomByPath(geometryOutputVertexes))
euclidean.translateVector3Path(geometryOutputVertexes, translation)
return geometryOutput
def getLighteningHoles(derivation, gearHolePaths, pitchRadius):
'Get cutout circles.'
if gearHolePaths != None:
if len(gearHolePaths) > 0:
return gearHolePaths
innerRadius = abs(pitchRadius) - derivation.dedendum
lighteningHoleOuterRadius = innerRadius - derivation.rimDedendum
shaftRimRadius = max(derivation.shaftRimRadius, (lighteningHoleOuterRadius) * (0.5 - math.sqrt(0.1875)))
lighteningHoleRadius = 0.5 * (lighteningHoleOuterRadius - derivation.shaftRimRadius)
if lighteningHoleRadius < derivation.lighteningHoleMinimumRadius:
return []
lighteningHoles = []
numberOfLighteningHoles = 3
polygonRadius = lighteningHoleOuterRadius - lighteningHoleRadius
rimDemiwidth = 0.5 * derivation.lighteningHoleMargin
axialMargin = getAxialMargin(lighteningHoleRadius, numberOfLighteningHoles, polygonRadius)
if axialMargin < rimDemiwidth:
while axialMargin < rimDemiwidth:
lighteningHoleRadius *= 0.999
if lighteningHoleRadius < derivation.lighteningHoleMinimumRadius:
return []
axialMargin = getAxialMargin(lighteningHoleRadius, numberOfLighteningHoles, polygonRadius)
else:
newNumberOfLighteningHoles = numberOfLighteningHoles
while axialMargin > rimDemiwidth:
numberOfLighteningHoles = newNumberOfLighteningHoles
newNumberOfLighteningHoles += 2
axialMargin = getAxialMargin(lighteningHoleRadius, newNumberOfLighteningHoles, polygonRadius)
sideAngle = 2.0 * math.pi / float(numberOfLighteningHoles)
startAngle = 0.0
for lighteningHoleIndex in xrange(numberOfLighteningHoles):
unitPolar = euclidean.getWiddershinsUnitPolar(startAngle)
lighteningHole = euclidean.getComplexPolygon(unitPolar * polygonRadius, lighteningHoleRadius, -13)
lighteningHoles.append(lighteningHole)
startAngle += sideAngle
return euclidean.getVector3Paths(lighteningHoles)
def getNewDerivation(elementNode):
'Get new derivation.'
return GearDerivation(elementNode)
def getOutputCylinder(
collarLength, derivation, elementNode, gearHolePaths, pitchRadius, teeth, twist, vector3GearProfile):
"Get extrude output for a cylinder gear."
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = [Vector3(), Vector3(0.0, 0.0, derivation.faceWidth)]
extrudeDerivation = extrude.ExtrudeDerivation(copyShallow)
negatives = []
positives = []
if twist != 0.0:
twistDegrees = math.degrees(twist)
extrudeDerivation.twistPathDefault = []
for complexPoint in derivation.helixPath:
extrudeDerivation.twistPathDefault.append(Vector3(complexPoint.real, twistDegrees * complexPoint.imag))
extrude.insertTwistPortions(extrudeDerivation, elementNode)
if derivation.operatingAngle != 180.0:
addBevelGear(derivation, extrudeDerivation, pitchRadius, positives, teeth, vector3GearProfile)
addCollarShaft(collarLength, derivation, elementNode, negatives, positives)
return extrude.getGeometryOutputByNegativesPositives(elementNode, negatives, positives)
if pitchRadius > 0:
extrude.addNegativesPositives(extrudeDerivation, negatives, vector3GearProfile, positives)
addLighteningHoles(derivation, gearHolePaths, negatives, pitchRadius, positives)
addCollarShaft(collarLength, derivation, elementNode, negatives, positives)
return extrude.getGeometryOutputByNegativesPositives(elementNode, negatives, positives)
if derivation.plateLength <= 0.0:
extrude.addNegativesPositives(extrudeDerivation, negatives, vector3GearProfile, positives)
return extrude.getGeometryOutputByNegativesPositives(elementNode, negatives, positives)
portionDirections = extrude.getSpacedPortionDirections(extrudeDerivation.interpolationDictionary)
outerGearProfile = vector3GearProfile[0]
outerLoopLists = extrude.getLoopListsByPath(extrudeDerivation, None, outerGearProfile, portionDirections)
addBottomLoop(-derivation.plateClearance, outerLoopLists[0])
geometryOutput = triangle_mesh.getPillarsOutput(outerLoopLists)
positives.append(geometryOutput)
innerLoopLists = extrude.getLoopListsByPath(extrudeDerivation, None, vector3GearProfile[1], portionDirections)
addBottomLoop(-derivation.plateClearance, innerLoopLists[0])
geometryOutput = triangle_mesh.getPillarsOutput(innerLoopLists)
negatives.append(geometryOutput)
connectionStart = Vector3(0.0, 0.0, -derivation.plateLength)
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = [connectionStart, Vector3(0.0, 0.0, -derivation.plateClearance)]
plateDerivation = extrude.ExtrudeDerivation(copyShallow)
extrude.addNegativesPositives(plateDerivation, negatives, [outerGearProfile], positives)
vector3LighteningHoles = getLighteningHoles(derivation, gearHolePaths, pitchRadius)
extrude.addNegativesPositives(plateDerivation, negatives, vector3LighteningHoles, positives)
addShaft(derivation, negatives, positives)
positiveOutput = triangle_mesh.getUnifiedOutput(positives)
annulusPlateOutput = {'difference' : {'shapes' : [positiveOutput] + negatives}}
if collarLength <= 0.0:
outputCylinder = solid.getGeometryOutputByManipulation(elementNode, annulusPlateOutput)
return getLiftedOutput(derivation, outputCylinder)
negatives = []
positives = []
connectionEnd = Vector3(0.0, 0.0, derivation.faceWidth + collarLength)
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = [Vector3(0.0, 0.0, -derivation.plateClearance), connectionEnd]
collarDerivation = extrude.ExtrudeDerivation(copyShallow)
addCollarShaftSetDerivation(collarDerivation, collarLength, derivation, elementNode, negatives, positives)
collarOutput = {'difference' : {'shapes' : positives + negatives}}
cylinderOutput = {'union' : {'shapes' : [annulusPlateOutput, collarOutput]}}
outputCylinder = solid.getGeometryOutputByManipulation(elementNode, cylinderOutput)
return getLiftedOutput(derivation, outputCylinder)
def getOutputRack(derivation, elementNode, vector3GearProfile):
"Get extrude output for a rack."
path = []
for complexPoint in derivation.helixPath:
point = Vector3(derivation.helixHeight * complexPoint.imag, 0.0, derivation.faceWidth * complexPoint.real)
path.append(point)
copyShallow = derivation.elementNode.getCopyShallow()
copyShallow.attributes['path'] = path
extrudeDerivation = extrude.ExtrudeDerivation(copyShallow)
negatives = []
positives = []
vector3RackProfiles = [vector3GearProfile]
if derivation.extraRackDemilength > 0.0:
yMaximum = -912345678.0
yMinimum = 912345678.0
for point in vector3GearProfile:
yMaximum = max(point.y, yMaximum)
yMinimum = min(point.y, yMinimum)
muchLessThanWidth = 0.01 * derivation.rackWidth
yMaximum += muchLessThanWidth
yMinimum -= muchLessThanWidth
extraRackLength = derivation.extraRackDemilength + derivation.extraRackDemilength
rackDemilengthPlus = derivation.rackDemilength + extraRackLength
leftNegative = [
Vector3(-derivation.rackDemilength, yMaximum),
Vector3(-derivation.rackDemilength, yMinimum),
Vector3(-rackDemilengthPlus, yMinimum),
Vector3(-rackDemilengthPlus, yMaximum)]
vector3RackProfiles.append(leftNegative)
rightNegative = [
Vector3(rackDemilengthPlus, yMaximum),
Vector3(rackDemilengthPlus, yMinimum),
Vector3(derivation.rackDemilength, yMinimum),
Vector3(derivation.rackDemilength, yMaximum)]
vector3RackProfiles.append(rightNegative)
addRackHoles(derivation, elementNode, vector3RackProfiles)
extrude.addNegativesPositives(extrudeDerivation, negatives, vector3RackProfiles, positives)
return extrude.getGeometryOutputByNegativesPositives(elementNode, negatives, positives)
def getPathOutput(creationFirst, derivation, elementNode, translation, vector3ComplementPaths, vector3PinionProfile):
"Get gear path output."
vector3PinionProfile = lineation.getPackedGeometryOutputByLoop(elementNode, lineation.SideLoop(vector3PinionProfile))
if creationFirst == 'p':
return vector3PinionProfile
packedGearGeometry = []
for vector3ComplementPath in vector3ComplementPaths:
sideLoop = lineation.SideLoop(vector3ComplementPath)
packedGearGeometry += lineation.getPackedGeometryOutputByLoop(elementNode, sideLoop)
if creationFirst == 'c':
return packedGearGeometry
euclidean.translateVector3Paths(packedGearGeometry, translation)
return vector3PinionProfile + packedGearGeometry
def getThicknessMultipliedPath(path, thicknessMultiplier):
"Get thickness multiplied path."
for pointIndex, point in enumerate(path):
path[pointIndex] = complex(point.real * thicknessMultiplier, point.imag)
return path
def getToothProfile(derivation, pitchRadius, teeth):
'Get profile for one tooth.'
if teeth < 0:
return getToothProfileAnnulus(derivation, pitchRadius, teeth)
if teeth == 0:
return getToothProfileRack(derivation)
return getToothProfileCylinder(derivation, pitchRadius, teeth)
def getToothProfileAnnulus(derivation, pitchRadius, teeth):
'Get profile for one tooth of an annulus.'
toothProfileHalf = []
toothProfileHalfCylinder = getToothProfileHalfCylinder(derivation, pitchRadius)
pitchRadius = -pitchRadius
innerRadius = pitchRadius - derivation.addendum
# tooth is multiplied by 1.02 because at around 1.01 for a 7/-23/20.0 test case, there is intersection since the paths are bending together
for point in getThicknessMultipliedPath(toothProfileHalfCylinder, 1.02 / derivation.toothThicknessMultiplier):
if abs(point) >= innerRadius:
toothProfileHalf.append(point)
profileFirst = toothProfileHalf[0]
profileSecond = toothProfileHalf[1]
firstMinusSecond = profileFirst - profileSecond
remainingAddendum = abs(profileFirst) - innerRadius
firstMinusSecond *= remainingAddendum / abs(firstMinusSecond)
extensionPoint = profileFirst + firstMinusSecond
if derivation.tipBevel > 0.0:
unitPolar = euclidean.getWiddershinsUnitPolar(2.0 / float(teeth) * math.pi)
mirrorPoint = complex(-extensionPoint.real, extensionPoint.imag) * unitPolar
bevelPath = getBevelPath(profileFirst, derivation.tipBevel, extensionPoint, mirrorPoint)
toothProfileHalf = bevelPath + toothProfileHalf
else:
toothProfileHalf.insert(0, extensionPoint)
profileLast = toothProfileHalf[-1]
profilePenultimate = toothProfileHalf[-2]
lastMinusPenultimate = profileLast - profilePenultimate
remainingDedendum = pitchRadius - abs(profileLast) + derivation.dedendum
lastMinusPenultimate *= remainingDedendum / abs(lastMinusPenultimate)
extensionPoint = profileLast + lastMinusPenultimate
if derivation.rootBevel > 0.0:
mirrorPoint = complex(-extensionPoint.real, extensionPoint.imag)
bevelPath = getBevelPath(profileLast, derivation.rootBevel, extensionPoint, mirrorPoint)
bevelPath.reverse()
toothProfileHalf += bevelPath
else:
toothProfileHalf.append(extensionPoint)
toothProfileAnnulus = euclidean.getMirrorPath(toothProfileHalf)
toothProfileAnnulus.reverse()
return toothProfileAnnulus
def getToothProfileCylinder(derivation, pitchRadius, teeth):
'Get profile for one tooth of a cylindrical gear.'
toothProfileHalfCylinder = getToothProfileHalfCylinder(derivation, pitchRadius)
toothProfileHalfCylinder = getThicknessMultipliedPath(toothProfileHalfCylinder, derivation.toothThicknessMultiplier)
toothProfileHalf = []
innerRadius = pitchRadius - derivation.dedendum
for point in toothProfileHalfCylinder:
if abs(point) >= innerRadius:
toothProfileHalf.append(point)
return getToothProfileCylinderByProfile(derivation, pitchRadius, teeth, toothProfileHalf)
def getToothProfileCylinderByProfile(derivation, pitchRadius, teeth, toothProfileHalf):
'Get profile for one tooth of a cylindrical gear.'
profileFirst = toothProfileHalf[0]
profileSecond = toothProfileHalf[1]
firstMinusSecond = profileFirst - profileSecond
remainingDedendum = abs(profileFirst) - pitchRadius + derivation.dedendum
firstMinusSecond *= remainingDedendum / abs(firstMinusSecond)
extensionPoint = profileFirst + firstMinusSecond
if derivation.rootBevel > 0.0:
unitPolar = euclidean.getWiddershinsUnitPolar(-2.0 / float(teeth) * math.pi)
mirrorPoint = complex(-extensionPoint.real, extensionPoint.imag) * unitPolar
bevelPath = getBevelPath(profileFirst, derivation.rootBevel, extensionPoint, mirrorPoint)
toothProfileHalf = bevelPath + toothProfileHalf
else:
toothProfileHalf.insert(0, extensionPoint)
if derivation.tipBevel > 0.0:
profileLast = toothProfileHalf[-1]
profilePenultimate = toothProfileHalf[-2]
mirrorPoint = complex(-profileLast.real, profileLast.imag)
bevelPath = getBevelPath(profilePenultimate, derivation.tipBevel, profileLast, mirrorPoint)
bevelPath.reverse()
toothProfileHalf = toothProfileHalf[: -1] + bevelPath
return euclidean.getMirrorPath(toothProfileHalf)
def getToothProfileHalfCylinder(derivation, pitchRadius):
'Get profile for half of a one tooth of a cylindrical gear.'
toothProfile=[]
# x = -y * tan(p) + 1
# x*x + y*y = (2-cos(p))^2
# y*y*t*t-2yt+1+y*y=4-4c-c*c
# y*y*(t*t+1)-2yt=3-4c-c*c
# y*y*(t*t+1)-2yt-3+4c-c*c=0
# a=tt+1
# b=-2t
# c=c(4-c)-3
a = derivation.tanPressure * derivation.tanPressure + 1.0
b = -derivation.tanPressure - derivation.tanPressure
cEnd = derivation.cosPressure * (4.0 - derivation.cosPressure) - 3.0
yEnd = (-b - math.sqrt(b*b - 4 * a * cEnd)) * 0.5 / a
yEnd *= derivation.pitchRadius / abs(pitchRadius)
yEnd -= derivation.clearance / abs(pitchRadius)
# to prevent intersections, yBegin is moved towards the base circle, giving a thinner tooth
yBegin = -yEnd
if pitchRadius > 0.0:
yBegin = 0.5 * derivation.sinPressure + 0.5 * yBegin
beginComplex = complex(1.0 - yBegin * derivation.tanPressure, yBegin)
endComplex = complex(1.0 - yEnd * derivation.tanPressure, yEnd)
endMinusBeginComplex = endComplex - beginComplex
wholeAngle = -abs(endMinusBeginComplex) / derivation.cosPressure
wholeAngleIncrement = wholeAngle / float(derivation.profileSurfaces)
stringStartAngle = abs(beginComplex - complex(1.0, 0.0)) / derivation.cosPressure
wholeDepthIncrementComplex = endMinusBeginComplex / float(derivation.profileSurfaces)
for profileIndex in xrange(derivation.profileSurfaces + 1):
contactPoint = beginComplex + wholeDepthIncrementComplex * float(profileIndex)
stringAngle = stringStartAngle + wholeAngleIncrement * float(profileIndex)
angle = math.atan2(contactPoint.imag, contactPoint.real) - stringAngle
angle += 0.5 * math.pi - derivation.quarterWavelength / abs(pitchRadius)
toothPoint = abs(contactPoint) * euclidean.getWiddershinsUnitPolar(angle) * abs(pitchRadius)
toothProfile.append(toothPoint)
return toothProfile
def getToothProfileRack(derivation):
'Get profile for one rack tooth.'
addendumSide = derivation.quarterWavelength - derivation.addendum * derivation.tanPressure
addendumComplex = complex(addendumSide, derivation.addendum)
dedendumSide = derivation.quarterWavelength + derivation.dedendum * derivation.tanPressure
dedendumComplex = complex(dedendumSide, -derivation.dedendum)
toothProfile = [dedendumComplex]
if derivation.rootBevel > 0.0:
mirrorPoint = complex(derivation.wavelength - dedendumSide, -derivation.dedendum)
toothProfile = getBevelPath(addendumComplex, derivation.rootBevel, dedendumComplex, mirrorPoint)
if derivation.tipBevel > 0.0:
mirrorPoint = complex(-addendumComplex.real, addendumComplex.imag)
bevelPath = getBevelPath(dedendumComplex, derivation.tipBevel, addendumComplex, mirrorPoint)
bevelPath.reverse()
toothProfile += bevelPath
else:
toothProfile.append(addendumComplex)
return euclidean.getMirrorPath(getThicknessMultipliedPath(toothProfile, derivation.toothThicknessMultiplier))
def processElementNode(elementNode):
"Process the xml element."
geometryOutput = getGeometryOutput(None, elementNode)
if geometryOutput.__class__ == list:
path.convertElementNode(elementNode, geometryOutput)
else:
solid.processElementNodeByGeometry(elementNode, geometryOutput)
class GearDerivation:
"Class to hold gear variables."
def __init__(self, elementNode):
'Set defaults.'
self.clearanceOverWavelength = evaluate.getEvaluatedFloat(0.1, elementNode, 'clearanceOverWavelength')
self.collarAddendumOverRadius = evaluate.getEvaluatedFloat(1.0, elementNode, 'collarAddendumOverRadius')
self.complementCollarLengthOverFaceWidth = evaluate.getEvaluatedFloat(
0.0, elementNode, 'complementCollarLengthOverFaceWidth')
self.copyShallow = elementNode.getCopyShallow()
self.creationType = evaluate.getEvaluatedString('both', elementNode, 'creationType')
self.creationTypeMenuRadioStrings = 'both complement pinion'.split()
self.elementNode = elementNode
self.faceWidth = evaluate.getEvaluatedFloat(10.0, elementNode, 'faceWidth')
self.helixAngle = evaluate.getEvaluatedFloat(0.0, elementNode, 'helixAngle')
self.helixType = evaluate.getEvaluatedString('basic', elementNode, 'helixType')
self.helixTypeMenuRadioStrings = 'basic herringbone parabolic'.split()
self.keywayRadiusOverRadius = evaluate.getEvaluatedFloat(0.5, elementNode, 'keywayRadiusOverRadius')
self.lighteningHoleMarginOverRimDedendum = evaluate.getEvaluatedFloat(
1.0, elementNode, 'lighteningHoleMarginOverRimDedendum')
self.lighteningHoleMinimumRadius = evaluate.getEvaluatedFloat(
1.0, elementNode, 'lighteningHoleMinimumRadius')
self.moveType = evaluate.getEvaluatedString('separate', elementNode, 'moveType')
self.moveTypeMenuRadioStrings = 'mesh none separate vertical'.split()
self.operatingAngle = evaluate.getEvaluatedFloat(180.0, elementNode, 'operatingAngle')
self.pinionCollarLengthOverFaceWidth = evaluate.getEvaluatedFloat(
0.0, elementNode, 'pinionCollarLengthOverFaceWidth')
self.plateClearanceOverLength = evaluate.getEvaluatedFloat(0.2, elementNode, 'plateClearanceOverLength')
self.plateLengthOverFaceWidth = evaluate.getEvaluatedFloat(0.5, elementNode, 'plateLengthOverFaceWidth')
self.pressureAngle = evaluate.getEvaluatedFloat(20.0, elementNode, 'pressureAngle')
self.profileSurfaces = evaluate.getEvaluatedInt(11, elementNode, 'profileSurfaces')
self.rackHoleBelowOverWidth = evaluate.getEvaluatedFloat(0.6, elementNode, 'rackHoleBelowOverWidth')
self.rackHoleRadiusOverWidth = evaluate.getEvaluatedFloat(0.0, elementNode, 'rackHoleRadiusOverWidth')
self.rackHoleStepOverWidth = evaluate.getEvaluatedFloat(1.0, elementNode, 'rackHoleStepOverWidth')
self.rackLengthOverRadius = evaluate.getEvaluatedFloat(math.pi + math.pi, elementNode, 'rackLengthOverRadius')
self.rackWidthOverFaceWidth = evaluate.getEvaluatedFloat(1.0, elementNode, 'rackWidthOverFaceWidth')
self.rimDedendumOverRadius = evaluate.getEvaluatedFloat(0.2, elementNode, 'rimDedendumOverRadius')
self.rootBevelOverClearance = evaluate.getEvaluatedFloat(0.5, elementNode, 'rootBevelOverClearance')
self.shaftDepthBottomOverRadius = evaluate.getEvaluatedFloat(0.0, elementNode, 'shaftDepthBottomOverRadius')
self.shaftDepthTopOverRadius = evaluate.getEvaluatedFloat(0.0, elementNode, 'shaftDepthOverRadius')
self.shaftRadiusOverPitchRadius = evaluate.getEvaluatedFloat(0.0, elementNode, 'shaftRadiusOverPitchRadius')
self.shaftSides = evaluate.getEvaluatedInt(4, elementNode, 'shaftSides')
self.teethComplement = evaluate.getEvaluatedInt(17, elementNode, 'teethComplement')
self.teethPinion = evaluate.getEvaluatedInt(7, elementNode, 'teeth')
totalTeethOverPinionTeeth = float(self.teethComplement + self.teethPinion) / float(self.teethPinion)
self.centerDistance = evaluate.getEvaluatedFloat(20.0 * totalTeethOverPinionTeeth, elementNode, 'centerDistance')
derivedPitchRadius = self.centerDistance / totalTeethOverPinionTeeth
self.pitchRadius = evaluate.getEvaluatedFloat(derivedPitchRadius, elementNode, 'pitchRadius')
self.tipBevelOverClearance = evaluate.getEvaluatedFloat(0.1, elementNode, 'tipBevelOverClearance')
# tooth multiplied by 0.99999 to avoid an intersection
self.toothThicknessMultiplier = evaluate.getEvaluatedFloat(0.99999, elementNode, 'toothThicknessMultiplier')
# Set derived variables.
self.wavelength = self.pitchRadius * 2.0 * math.pi / float(self.teethPinion)
self.clearance = self.wavelength * self.clearanceOverWavelength
self.clearance = evaluate.getEvaluatedFloat(self.clearance, elementNode, 'clearance')
self.complementCollarLength = self.faceWidth * self.complementCollarLengthOverFaceWidth
self.complementCollarLength = evaluate.getEvaluatedFloat(self.complementCollarLength, elementNode, 'complementCollarLength')
self.gearHolePaths = evaluate.getTransformedPathsByKey([], elementNode, 'gearHolePaths')
self.pinionCollarLength = self.faceWidth * self.pinionCollarLengthOverFaceWidth
self.pinionCollarLength = evaluate.getEvaluatedFloat(self.pinionCollarLength, elementNode, 'pinionCollarLength')
self.plateLength = self.faceWidth * self.plateLengthOverFaceWidth
self.plateLength = evaluate.getEvaluatedFloat(self.plateLength, elementNode, 'plateLength')
self.plateClearance = self.plateLength * self.plateClearanceOverLength
self.plateClearance = evaluate.getEvaluatedFloat(self.plateClearance, elementNode, 'plateClearance')
self.rackLength = self.pitchRadius * self.rackLengthOverRadius
self.rackLength = evaluate.getEvaluatedFloat(self.rackLength, elementNode, 'rackLength')
self.rackDemilength = 0.5 * self.rackLength
self.rackWidth = self.faceWidth * self.rackWidthOverFaceWidth
self.rackWidth = evaluate.getEvaluatedFloat(self.rackWidth, elementNode, 'rackWidth')
self.rimDedendum = self.pitchRadius * self.rimDedendumOverRadius
self.rimDedendum = evaluate.getEvaluatedFloat(self.rimDedendum, elementNode, 'rimDedendum')
self.rootBevel = self.clearance * self.rootBevelOverClearance
self.rootBevel = evaluate.getEvaluatedFloat(self.rootBevel, elementNode, 'rootBevel')
self.shaftRadius = self.pitchRadius * self.shaftRadiusOverPitchRadius
self.shaftRadius = evaluate.getEvaluatedFloat(self.shaftRadius, elementNode, 'shaftRadius')
self.collarAddendum = self.shaftRadius * self.collarAddendumOverRadius
self.collarAddendum = evaluate.getEvaluatedFloat(self.collarAddendum, elementNode, 'collarWidth')
self.keywayRadius = self.shaftRadius * self.keywayRadiusOverRadius
self.keywayRadius = lineation.getFloatByPrefixBeginEnd(elementNode, 'keywayRadius', 'keywayDiameter', self.keywayRadius)
self.lighteningHoleMargin = self.rimDedendum * self.lighteningHoleMarginOverRimDedendum
self.lighteningHoleMargin = evaluate.getEvaluatedFloat(
self.lighteningHoleMargin, elementNode, 'lighteningHoleMargin')
self.rackHoleBelow = self.rackWidth * self.rackHoleBelowOverWidth
self.rackHoleBelow = evaluate.getEvaluatedFloat(self.rackHoleBelow, elementNode, 'rackHoleBelow')
self.rackHoleRadius = self.rackWidth * self.rackHoleRadiusOverWidth
self.rackHoleRadius = lineation.getFloatByPrefixBeginEnd(elementNode, 'rackHoleRadius', 'rackHoleDiameter', self.rackHoleRadius)
self.rackHoleStep = self.rackWidth * self.rackHoleStepOverWidth
self.rackHoleStep = evaluate.getEvaluatedFloat(self.rackHoleStep, elementNode, 'rackHoleStep')
self.shaftDepthBottom = self.shaftRadius * self.shaftDepthBottomOverRadius
self.shaftDepthBottom = evaluate.getEvaluatedFloat(self.shaftDepthBottom, elementNode, 'shaftDepthBottom')
self.shaftDepthTop = self.shaftRadius * self.shaftDepthTopOverRadius
self.shaftDepthTop = evaluate.getEvaluatedFloat(self.shaftDepthTop, elementNode, 'shaftDepthTop')
self.shaftPath = evaluate.getTransformedPathByKey([], elementNode, 'shaftPath')
if len(self.shaftPath) < 3:
self.shaftPath = shaft.getShaftPath(self.shaftDepthBottom, self.shaftDepthTop, self.shaftRadius, -self.shaftSides)
self.tipBevel = self.clearance * self.tipBevelOverClearance
self.tipBevel = evaluate.getEvaluatedFloat(self.tipBevel, elementNode, 'tipBevel')
# Set derived values.
self.helixRadian = math.radians(self.helixAngle)
if self.teethComplement <= 0.0 and self.operatingAngle != 180.0:
print('Warning, an operatingAngle other than 180 degrees can only work with a positive number of gear teeth.')
print('Therefore the operatingAngle will be reset to 180 degrees.')
self.operatingAngle = 180.0
self.tanHelix = math.tan(self.helixRadian)
self.helixHeight = self.tanHelix * self.faceWidth
self.operatingRadian = math.radians(self.operatingAngle)
self.pitchRadiusComplement = self.pitchRadius * float(self.teethComplement) / float(self.teethPinion)
self.pressureRadian = math.radians(self.pressureAngle)
self.cosPressure = math.cos(self.pressureRadian)
self.sinPressure = math.sin(self.pressureRadian)
self.tanPressure = math.tan(self.pressureRadian)
self.halfWavelength = 0.5 * self.wavelength
self.helixPath = euclidean.getComplexPath(evaluate.getTransformedPathByKey([], elementNode, 'helixPath'))
if len(self.helixPath) < 1:
self.helixPath = getHelixComplexPath(self, elementNode)
self.quarterWavelength = 0.25 * self.wavelength
self.shaftRimRadius = self.shaftRadius + self.collarAddendum
self.toothProfileHalf = getToothProfileHalfCylinder(self, self.pitchRadius)
self.toothProfileHalf = getThicknessMultipliedPath(self.toothProfileHalf, self.toothThicknessMultiplier)
self.addendum = self.toothProfileHalf[-1].imag - self.pitchRadius
self.dedendum = abs(self.toothProfileHalf[-1]) - self.pitchRadius + self.clearance
self.pinionToothProfile = getToothProfileCylinderByProfile(self, self.pitchRadius, self.teethPinion, self.toothProfileHalf)