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()