Skip to content
Snippets Groups Projects
mill.py 4.01 KiB
Newer Older
"""
Small showcase posted at http://www.youtube.com/watch?v=KUv26xlh89I,
in response to pfc3d's http://www.youtube.com/watch?v=005rdDBoe4w.
Physical correctness is not the focus, the geometry and similar look is.

You can take this file as instruction on how to build parametric surfaces,
and how to make videos as well.
"""
from yade import pack
from numpy import linspace
# geometry parameters
bumpNum = 20
bumpHt, bumpTipAngle = 0.07, 60 * pi / 180
millRad, millDp = 1, 1  # radius and depth (cylinder length) of the mill
sphRad, sphRadFuzz = 0.03, .8  # mean radius and relative fuzz of the radius (random, uniformly distributed between sphRad*(1-.5*sphRadFuzz)…sphRad*(1+.5*sphRadFuzz))
dTheta = pi / 24  # circle division angle

###
### mill geometry (parameteric)
###
bumpPeri = 2 * bumpHt * tan(.5 * bumpTipAngle)  # length of a bump on the perimeter of the mill
bumpAngle = bumpPeri / millRad  # angle of one bump from the axis of the mill
interBumpAngle = 2 * pi / bumpNum
bumpRad = millRad - bumpHt
pts = []
thMin = 0
for i in range(0, bumpNum):
	thMin += interBumpAngle
	thMax = thMin + interBumpAngle - bumpAngle
	thTip = thMax + .5 * bumpAngle
	# the circular parts spanning from thMin to thMax
	n = int(interBumpAngle / dTheta)
	for th0 in linspace(thMin, thMax, n, endpoint=True):
		pts.append(Vector3(-.5 * millDp, millRad * cos(th0), millRad * sin(th0)))
	pts.append(Vector3(-.5 * millDp, bumpRad * cos(thTip), bumpRad * sin(thTip)))
# make the second contour, just shifted by millDp; ppts contains both
ppts = [pts, [p + Vector3(millDp, 0, 0) for p in pts]]
mill = pack.sweptPolylines2gtsSurface(ppts, threshold=.01 * min(dTheta * millRad, bumpHt))  #,capStart=True,capEnd=True)
millIds = O.bodies.append(pack.gtsSurface2Facets(mill, color=(1, 0, 1), wire=False))  # add triangles, save their ids
# make the caps less comfortably, but looking better as two triangle couples over the mill
mrs2 = millRad * sqrt(2)
cap1, cap2 = [Vector3(0, 0, mrs2), Vector3(0, -mrs2, 0),
              Vector3(0, 0, -mrs2)], [Vector3(0, 0, mrs2), Vector3(0, 0, -mrs2), Vector3(0, mrs2, 0)]  # 2 triangles at every side
for xx in -.5 * millDp, .5 * millDp:
	millIds += O.bodies.append(
	        [facet([p + Vector3(xx, 0, 0) for p in cap1], color=(0, 0, 0)),
	         facet([p + Vector3(xx, 0, 0) for p in cap2], color=(0, 0, 0))]
	)

# define domains for initial cloud of red and blue spheres
packHt = .8 * millRad  # size of the area
bboxes = [
        (Vector3(-.5 * millDp, -.5 * packHt, -.5 * packHt), Vector3(.5 * millDp, 0, .5 * packHt)),
        (Vector3(-.5 * millDp, 0, -.5 * packHt), Vector3(.5 * millDp, .5 * packHt, .5 * packHt))
]
colors = (1, 0, 0), (0, 0, 1)
for i in (0, 1):  # red and blue spheres
	sp = pack.SpherePack()
	bb = bboxes[i]
	vol = (bb[1][0] - bb[0][0]) * (bb[1][1] - bb[0][1]) * (bb[1][2] - bb[0][2])
	sp.makeCloud(bb[0], bb[1], sphRad, sphRadFuzz, int(.25 * vol / ((4. / 3) * pi * sphRad**3)), False)
	O.bodies.append([sphere(s[0], s[1], color=colors[i]) for s in sp])
print("Numer of grains", len(O.bodies) - len(millIds))
O.dt = .5 * PWaveTimeStep()
O.engines = [
        ForceResetter(),
        InsertionSortCollider([Bo1_Sphere_Aabb(), Bo1_Facet_Aabb()]),
        InteractionLoop(
                [Ig2_Sphere_Sphere_ScGeom(), Ig2_Facet_Sphere_ScGeom()],
                [Ip2_FrictMat_FrictMat_FrictPhys()],
                [Law2_ScGeom_FrictPhys_CundallStrack()],
        ),
        NewtonIntegrator(damping=.3, gravity=(0, 0, -50)),
        RotationEngine(rotateAroundZero=True, zeroPoint=(0, 0, 0), rotationAxis=(1, 0, 0), angularVelocity=-20, ids=millIds),
        #SnapshotEngine(iterPeriod=30,fileBase='/tmp/mill-',viewNo=0,label='snapshooter'),
        #VTKRecorder(iterPeriod=100,recorders=['all'],fileName='/tmp/millVTK-')
v = qt.View()
v.eyePosition = (3, .8, .96)
v.upVector = (-.4, -.4, .8)
v.viewDir = (-.9, -.25, -.3)
v.axes = True
v.sceneRadius = 1.9
#encodeVideoFromFrames(snapshooter['savedSnapshots'],out='/tmp/mill.ogg',fps=30)