chainDriveExample.py

You can view and download this file on Github: chainDriveExample.py

  1#+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2# This is an EXUDYN example
  3#
  4# Details:  Test of a chain drive for ContactCurveCircles
  5#
  6# Author:   Johannes Gerstmayr
  7# Date:     2024-11-01
  8#
  9# Copyright:This file is part of Exudyn. Exudyn is free software. You can redistribute it and/or modify it under the terms of the Exudyn license. See 'LICENSE.txt' for more details.
 10#
 11#+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 12
 13import exudyn as exu
 14from exudyn.utilities import *
 15import exudyn.graphics as graphics
 16import numpy as np
 17
 18#**function: unique drawing for chain plate; reference position is at first roller;
 19#link is oriented into x-axis, thickness in z-axis, height in y-axis (height of rectangular part);
 20#zOff is the center of the thickness of the plate
 21#returns a single graphicsData object
 22def GraphicsChainPlate(lengthLink, heightLink, thicknessLink, radiusLink,
 23                       pOff=[0,0,0], color=[-1,-1,-1,-1], nTiles=16,
 24                       addEdges = False, addFaces=True):
 25    if heightLink > 2*radiusLink:
 26        raise ValueError('GraphicsChainPlate: height must by <= 2*radius!')
 27    from numpy import sin, cos, arcsin
 28
 29    gList = []
 30    vertices = []
 31    segments = []
 32    lastIndex = 2*(nTiles+1)-1
 33    phiStart = arcsin(0.5*heightLink/radiusLink)
 34
 35    #check if rectangular section can be drawn:
 36    if lengthLink - 2*cos(phiStart)*radiusLink < 0:
 37        phiStart = pi/2
 38
 39    phiEnd = 2*pi-phiStart
 40    phiRange = phiEnd-phiStart
 41
 42    for i in range(nTiles+1):
 43        phi = i/nTiles*phiRange+phiStart
 44        vertices.append([radiusLink*cos(phi),radiusLink*sin(phi)])
 45
 46    for i in range(nTiles+1):
 47        phi = i/nTiles*phiRange+pi+phiStart
 48        vertices.append([radiusLink*cos(phi)+lengthLink,radiusLink*sin(phi)])
 49
 50    segments = np.vstack((np.arange(2*(nTiles+1)),
 51                          np.arange(2*(nTiles+1))+1)).T
 52    segments[-1,-1] = 0
 53
 54    return graphics.SolidExtrusion(vertices, segments.tolist(), thicknessLink, pOff=np.array(pOff)-[0,0,0.5*thicknessLink],
 55                            color=color,addEdges=addEdges,addFaces=addFaces)
 56
 57
 58#**function: unique drawing for chain plate; reference position is at first roller;
 59#link is oriented into x-axis, thickness in z-axis, height in y-axis (height of rectangular part);
 60#zOff is the center of the thickness of the plate
 61#returns a single graphicsData object
 62def GraphicsChainLink(lengthLink, heightLink, thicknessLink, radiusLink, innerWidthLink,
 63                      radiusRoller, radiusPin, withPin, pOff=[0,0,0],
 64                      colorLink=graphics.color.grey, colorRoller=[0.7,0.7,0.8,1], colorPin=graphics.color.darkgrey,
 65                      nTiles=16, addEdges = False, addFaces=True):
 66
 67    gcl = []
 68    gcl += [graphics.Cylinder([0,0,-0.5*(withPin)],[0,0,withPin], radius=radiusPin,
 69                              color=colorRoller, nTiles=nTiles, addEdges=addEdges, addFaces=addFaces)]
 70    gcl += [GraphicsChainPlate(lengthLink, heightLink, thicknessLink, radiusLink,
 71                               pOff=[0,0,0.5*(innerWidthLink+thicknessLink)], color=colorLink,
 72                               nTiles=nTiles, addEdges=addEdges, addFaces=addFaces)]
 73    gcl += [GraphicsChainPlate(lengthLink, heightLink, thicknessLink, radiusLink,
 74                               pOff=[0,0,-0.5*(innerWidthLink+thicknessLink)], color=colorLink,
 75                               nTiles=nTiles, addEdges=addEdges, addFaces=addFaces)]
 76    gcl += [graphics.Cylinder([0,0,-0.5*(innerWidthLink)],[0,0,innerWidthLink], radius=radiusRoller,
 77                              color=colorRoller, nTiles=nTiles, addEdges=addEdges, addFaces=addFaces)]
 78    return gcl
 79
 80#unique drawing for roller chain link:
 81
 82
 83useGraphics = True #without test
 84SC = exu.SystemContainer()
 85mbs = SC.AddSystem()
 86
 87pi = np.pi
 88mass = 1
 89nTeeth = 16
 90nLinks3 = 7 #free links between sprockets
 91phiTooth = 2*pi/nTeeth
 92rRoller = 0.005
 93lLink = 0.03                                #Length of link; l = 2*r*sin(phi/2)
 94wLink = 0.012                                #inner with of link
 95tpLink = 0.002                              #thickness of plate
 96hpLink = 0.012                              #height of plate
 97rpLink = 1.5*rRoller                        #radius of plate
 98rSprocketPitch = lLink/(2*sin(phiTooth/2))  #radius to center of holes in gear, given by chain link length and nTeeth!
 99rSprocketInner = rSprocketPitch - rRoller
100wSprocket = wLink*0.95
101addSecondWheel = True
102contactStiffness = 1e4
103contactDamping = 1e1
104
105exu.Print('Chain geometry:')
106exu.Print('  lLink  =',lLink)
107exu.Print('  rRoller=',rRoller)
108exu.Print('  rPitch =',rSprocketPitch)
109exu.Print('  rInner =',rSprocketPitch)
110
111tEnd = 10     #end time of simulation
112stepSize = 1e-5
113g = 9.81
114
115gGround = graphics.CheckerBoard(point = [0,0,-wSprocket*5], size = 1)
116nGround = mbs.AddNode(NodePointGround())
117mCoordinateGround = mbs.AddMarker(MarkerNodeCoordinate(nodeNumber=nGround, coordinate=0))
118
119oGround = mbs.CreateGround(referencePosition=[0,0,0],graphicsDataList=[gGround])
120mGround = mbs.AddMarker(MarkerBodyRigid(bodyNumber=oGround))
121
122#ground node
123gListSprocket = [graphics.Cylinder([0,0,-0.5*wSprocket],[0,0,wSprocket],
124                 radius=rSprocketPitch, color=graphics.color.grey, nTiles=64)]
125
126pList = []
127nSprocket = nTeeth*4 #8
128for i in range(nSprocket):
129    angle = (i-0.5)/(nSprocket)*2*pi
130    r = rSprocketInner + 2*rRoller*(i%4 > 1)
131    x =-r*cos(angle)
132    y =-r*sin(angle)
133    pList.append([x,y])
134
135segments = np.vstack((np.arange(nSprocket),
136                        np.arange(nSprocket)+1)).T
137segments[-1,-1] = 0
138segmentsData = np.zeros((nSprocket,4))
139segmentsData[:,0:2] = pList
140segmentsData[:,2:4] = np.roll(pList,2) #roll is element wise on rows and columns, therefore 2>shift one row
141
142print(segmentsData[:5,:])
143
144gListSprocket = [
145    graphics.SolidExtrusion(pList, segments.tolist(), wSprocket, pOff=[0,0,-0.5*wSprocket],
146                            color=graphics.color.lightgrey,addEdges=True,addFaces=True),
147    graphics.Cylinder(pAxis=[0,0,-wSprocket], vAxis=[0,0,2*wSprocket], radius = rRoller, color=graphics.color.grey),
148    graphics.Brick(size=[0.5*wSprocket,wSprocket*2,wSprocket*1.05], color=graphics.color.red)
149]
150
151
152
153#sprocket wheel:
154sprocket = mbs.CreateRigidBody(referencePosition=[0,0,0],
155                                inertia=InertiaCylinder(density=100,length=wSprocket,outerRadius=rSprocketPitch,axis=2),
156                                gravity = [0,-g,0],
157                                graphicsDataList=gListSprocket,
158                                create2D=True
159                                )
160mSprocket = mbs.AddMarker(MarkerBodyRigid(bodyNumber=sprocket))
161mbs.AddObject(RevoluteJoint2D(markerNumbers=[mGround, mSprocket],
162                                visualization=VRevoluteJoint2D(drawSize=rRoller)) )
163
164#add prescribed velocity:
165if True:
166    def UFvelocityDrive(mbs, t, itemNumber, lOffset): #time derivative of UFoffset
167        vStart = 0
168        vEnd = pi
169        return SmoothStep(t, 0, 0.5, vStart, vEnd)
170
171    nodeSprocket = mbs.GetObject(sprocket)['nodeNumber']
172    mCoordinateWheel = mbs.AddMarker(MarkerNodeCoordinate(nodeNumber=nodeSprocket, coordinate=2))
173    velControl = mbs.AddObject(CoordinateConstraint(markerNumbers=[mCoordinateGround, mCoordinateWheel],
174                                        velocityLevel=True, offsetUserFunction_t= UFvelocityDrive,
175                                        visualization=VCoordinateConstraint(show = False)))#UFvelocityDrive
176else:
177    mbs.AddLoad(Torque(markerNumber=mSprocket, loadVector=[0,0,-0.1]))
178
179sVel1 = mbs.AddSensor(SensorBody(bodyNumber=sprocket, storeInternal=True, outputVariableType=exu.OutputVariableType.AngularVelocity))
180
181if addSecondWheel:
182    oGround2 = mbs.CreateGround(referencePosition=[0,-nLinks3*lLink,0],graphicsDataList=[])
183    mGround2 = mbs.AddMarker(MarkerBodyRigid(bodyNumber=oGround2))
184    sprocket2 = mbs.CreateRigidBody(referencePosition=[0,-nLinks3*lLink,0],
185                                    inertia=InertiaCylinder(density=100,length=wSprocket,outerRadius=rSprocketPitch,axis=2),
186                                    gravity = [0,-g,0],
187                                    graphicsDataList=gListSprocket,
188                                    create2D=True
189                                    )
190    # nn = mbs.GetObject(sprocket)['nodeNumber']
191    mSprocket2 = mbs.AddMarker(MarkerBodyRigid(bodyNumber=sprocket2))
192    mbs.AddObject(RevoluteJoint2D(markerNumbers=[mGround2, mSprocket2],
193                                    visualization=VRevoluteJoint2D(drawSize=rRoller)) )
194
195    dTSD = 0.01
196    mbs.AddObject(TorsionalSpringDamper(markerNumbers = [mGround, mSprocket2], damping = dTSD))
197    #mbs.AddLoad(Torque(markerNumber=mSprocket2, loadVector=[0,0,0.2]))
198    sVel2 = mbs.AddSensor(SensorBody(bodyNumber=sprocket2, storeInternal=True, outputVariableType=exu.OutputVariableType.AngularVelocity))
199
200
201#graphics for chain link:
202gChainLink1 = GraphicsChainLink(lLink, hpLink, tpLink, rpLink, wLink,
203                      rRoller, 0.3*rRoller, wLink+4.5*tpLink, pOff=[0,0,0],
204                      nTiles=32, addEdges = True)
205gChainLink2 = GraphicsChainLink(lLink, hpLink, tpLink, rpLink, wLink+tpLink*2,
206                      rRoller, 0.3*rRoller, wLink+4.5*tpLink, pOff=[0,0,0],
207                      nTiles=32, addEdges = True)
208
209#create geometry of hanging chain:
210posList = []
211rotList = []
212for i in range(nLinks3):
213    x=-rSprocketPitch
214    y=-(nLinks3-i)*lLink
215    posList.append([x,y])
216    rotList.append(pi/2)
217
218for i in range(int(nTeeth/2)):
219    angle = i/(nTeeth/2)*pi
220    x=-rSprocketPitch*cos(angle)
221    y= rSprocketPitch*sin(angle)
222    posList.append([x,y])
223    rotList.append(pi/2-angle-0.5/(nTeeth/2)*pi)
224
225for i in range(nLinks3):
226    x= rSprocketPitch
227    y=-(i+0)*lLink
228    posList.append([x,y])
229    rotList.append(-pi/2)
230
231for i in range(int(nTeeth/2)):
232    angle = -(i+0)/(nTeeth/2)*pi
233    x= rSprocketPitch*cos(angle)
234    y= rSprocketPitch*sin(angle)
235    posList.append([x,y-nLinks3*lLink])
236    rotList.append(-pi/2+angle-0.5/(nTeeth/2)*pi)
237
238oLinksList = []
239mLinksList = []
240
241prevMarker = None
242firstMarker = None
243for i in range(len(posList)):
244    pos = posList[i]
245    #note: in 2D case, the reference point = COM, which is not correct for chain; for correct representation, move ref. point to COM of link!
246    link = mbs.CreateRigidBody(referencePosition=pos+[0],
247                               inertia=InertiaCuboid(1000,sideLengths=[lLink,wLink,wLink]),
248                               referenceRotationMatrix=RotationMatrixZ(rotList[i]),
249                               gravity = [0,-g,0],
250                               graphicsDataList=gChainLink1 if i%2==0 else gChainLink2,
251                               create2D=True
252                               )
253    # nn = mbs.GetObject(sprocket)['nodeNumber']
254    oLinksList += [link]
255    mLinksList += [mbs.AddMarker(MarkerBodyRigid(bodyNumber=link))]
256
257    if prevMarker is not None:
258        mbs.AddObject(RevoluteJoint2D(markerNumbers=[prevMarker, mLinksList[-1]],
259                                      visualization=VRevoluteJoint2D(drawSize=rRoller)) )
260    else:
261        firstMarker = mLinksList[-1]
262
263    prevMarker = mbs.AddMarker(MarkerBodyRigid(bodyNumber=link, localPosition=[lLink,0,0]))
264
265#close chain:
266mbs.AddObject(RevoluteJoint2D(markerNumbers=[prevMarker, firstMarker],
267                                visualization=VRevoluteJoint2D(drawSize=rRoller)) )
268
269
270if False: #put one link to ground:
271    posLast = mbs.GetMarkerOutput(prevMarker, variableType=exu.OutputVariableType.Position,
272                                  configuration=exu.ConfigurationType.Reference)
273
274    oGround2 = mbs.CreateGround(referencePosition=posLast)
275    mGround2 = mbs.AddMarker(MarkerBodyRigid(bodyNumber=oGround2))
276    mbs.AddObject(RevoluteJoint2D(markerNumbers=[prevMarker, mGround2],
277                                  visualization=VRevoluteJoint2D(drawSize=rRoller)) )
278
279nGenericData = mbs.AddNode(NodeGenericData(initialCoordinates=[-1,0,0]*nSprocket,
280                                           numberOfDataCoordinates=3*nSprocket))
281mbs.AddObject(ObjectContactCurveCircles(markerNumbers=[mSprocket]+mLinksList, nodeNumber=nGenericData,
282                                        circlesRadii=[rRoller]*len(mLinksList), segmentsData=exu.MatrixContainer(segmentsData),
283                                        contactStiffness=contactStiffness, contactDamping=contactDamping))
284if addSecondWheel:
285    nGenericData2 = mbs.AddNode(NodeGenericData(initialCoordinates=[-1,0,0]*nSprocket,
286                                               numberOfDataCoordinates=3*nSprocket))
287    mbs.AddObject(ObjectContactCurveCircles(markerNumbers=[mSprocket2]+mLinksList, nodeNumber=nGenericData2,
288                                            circlesRadii=[rRoller]*len(mLinksList), segmentsData=exu.MatrixContainer(segmentsData),
289                                            contactStiffness=contactStiffness, contactDamping=contactDamping))
290
291
292mbs.Assemble()
293
294simulationSettings = exu.SimulationSettings()
295simulationSettings.solutionSettings.writeSolutionToFile = True
296simulationSettings.solutionSettings.solutionWritePeriod = 0.01
297simulationSettings.solutionSettings.sensorsWritePeriod = stepSize  #output interval
298simulationSettings.timeIntegration.numberOfSteps = int(tEnd/stepSize)
299simulationSettings.timeIntegration.endTime = tEnd
300#simulationSettings.timeIntegration.simulateInRealtime = True
301# simulationSettings.timeIntegration.discontinuous.iterationTolerance = 1e-2
302# simulationSettings.timeIntegration.discontinuous.useRecommendedStepSize = False
303
304simulationSettings.linearSolverType = exu.LinearSolverType.EigenSparse
305simulationSettings.timeIntegration.newton.useModifiedNewton = True
306#simulationSettings.timeIntegration.generalizedAlpha.spectralRadius = 1
307
308simulationSettings.displayStatistics = True
309simulationSettings.timeIntegration.verboseMode = 1
310
311SC.visualizationSettings.window.renderWindowSize=[1600,2000]
312SC.visualizationSettings.openGL.multiSampling=4
313#SC.visualizationSettings.openGL.facesTransparent=True
314SC.visualizationSettings.openGL.shadow=0.3
315SC.visualizationSettings.loads.show = False
316
317exu.StartRenderer()              #start graphics visualization
318mbs.WaitForUserToContinue()    #wait for pressing SPACE bar to continue
319
320#start solver:
321mbs.SolveDynamic(simulationSettings)
322
323SC.WaitForRenderEngineStopFlag()#wait for pressing 'Q' to quit
324exu.StopRenderer()               #safely close rendering window!
325
326#plot results:
327if False:
328    mbs.PlotSensor([sVel1,sVel2], components=[2,2], closeAll=True)
329    import matplotlib.pyplot as plt
330    plt.show(block=True) #for figures to stay open at end of plot routines
331
332#+++++++++++++++++++++++++++++++++++++++++++++++++++++
333
334# mbs.SolutionViewer()