# This is our primary file from which we analyze heat distribution in our system import math import numpy as np #import fdsreader #from fdsreader.bndf.utils import sort_patches_cartesian import matplotlib.pyplot as plt #import pandas as pd import matplotlib import matplotlib.animation as animation import random plt.style.use('_mpl-gallery-nogrid') # for rectilinear model: # Static position model defines grid as units of one with "adjacent" # being defined as within a 2 by 2 cube centered on the particle in question # Hexagonal Spacial: # Static positions defined as hexagonally offset # This ensures the distance from each particle to # another is the same for all close particles changeFactor = 1 #this is to make it easier to change the space between particles #This affects not only the placing but also the parameters for adjacency and class Particle(object): # change __init__ structure to 3d def __init__(self,xPos,yPos,zPos,temp): #these distances are in mm self.x = xPos self.y = yPos self.z = zPos self.t = temp self.Sx = 0 self.Sy = 0 self.Sz = 0 #angular velocities self.wx = 0 self.wy = 0 self.wz = 0 self.bondingGrade = -1 def changeTemp(self,newTemp): self.t = newTemp def changeX(self,newX): self.x = newX def changeY(self,newY): self.y = newY def changeZ(self,newZ): self.z = newZ def changeVX(self,newXVel): self.Sx = newXVel def changeVY(self,newYVel): self.Sy = newYVel def changeVZ(self,newZVel): self.Sz = newZVel def changeWX(self,newXVel): self.wx = newXVel def changeWY(self,newYVel): self.wy = newYVel def changeWZ(self,newZVel): self.wz = newZVel def setGrade(self,grade): self.bondingGrade = grade def isPlate(self): return False class Plate(Particle): def __init__(self,xPos,yPos,zPos): self.x = xPos self.y = yPos self.z = zPos self.t = plateTemp #velocities self.Sx = 0 self.Sy = 0 self.Sz = 0 #angular velocities self.wx = 0 self.wy = 0 self.wz = 0 self.bondingGrade = 0 #operates under the assumption of an unmoving plate of constant temperature def changeTemp(self,newTemp): self.t = plateTemp def changeVX(self,newXVel): self.Sx = 0 def changeVY(self,newYVel): self.Sy = 0 def changeVZ(self,newZVel): self.Sz = 0 def changeWX(self,newXVel): self.wx = 0 def changeWY(self,newYVel): self.wy = 0 def changeWZ(self,newZVel): self.wz = 0 def setGrade(self,grade): self.bondingGrade = 0 def isPlate(self): return True class Extruder(Particle): pass class Material(Particle): pass #set up sim #spacing is 1 unit for now iLength = 4 kLength = 4 jLength = 2 npS = 3 #num per side basepS = 4 # num per side of base #constriction (len = height) heightChanges = np.array(["I","O","I"]) height = len(heightChanges) #doesn't include base plateTemp = 50 #normal print bed temp bondingTemp = 68.5 #glass point extrusionTemp = 110 # C rateOfEx = 3 #(particle per second) timeFrame = 20 #seconds frameNum = 100000 visc = 1287000 #viscosity N/m^2 (Pa) Area = .000001 #area of side m^2 thermalExpansion = 4.17*10**(-4) #% per C TD = 5.8244*10**(-7) speedCap = .02 # m/s so 20mm/s forceCap = 100 #F 0.700348550209857 E = 17.5/0.672475 #base energy of bond (in Joules) such that the max force is m = .00000124 #kilograms systemTime = 0.0 #for reference whenever an update is made to the system the time should be adjusted repeating = False ''' #just some particles particleList = np.array([Particle(1,0,0,100),Particle(0,0,0,100),Particle(2,0,0,100),Particle(4,1,0,100),Particle(4,0,0,100)]) particleList[1].changeVY(.5) ''' #Make a rectilinear set (rudimentary) ''' particleList = np.array([None]*(iLength*kLength*jLength+(iLength+2)*(kLength+2))) particleListHist = np.array([[None]*(iLength*kLength*jLength+(iLength+2)*(kLength+2))]*(frameNum+1)) i=0 while(ilen(particleList)): particleNum = len(particleList) while(kFmax): Fmax=F changeInVX[adjacentIndices[i]] -= F/m*(particleActive.x-particle.x)/dist changeInVX[ind] += F/m*(particleActive.x-particle.x)/dist changeInVY[adjacentIndices[i]] -= F/m*(particleActive.y-particle.y)/dist changeInVY[ind] += F/m*(particleActive.y-particle.y)/dist changeInVZ[adjacentIndices[i]] -= F/m*(particleActive.z-particle.z)/dist changeInVZ[ind] += F/m*(particleActive.z-particle.z)/dist i+=1 #Torque and Angular Momentum d = [(particleActive.x-particle.x)/1000,(particleActive.y-particle.y)/1000,(particleActive.z-particle.z)/1000] #distance vector v1 = [particleActive.Sx-particle.Sx,particleActive.Sy-particle.Sy,particleActive.Sz-particle.Sz] #bare velocity difference wa =[particleActive.wx,particleActive.wy,particleActive.wz] #angular velocities wh =[particle.wx,particle.wy,particle.wz] r1 = [d[0]/2,d[1]/2,d[2]/2] #'radius' vector of home particle r2 = [-r1[0],-r1[1],-r1[2]] #'radius' vector of active particle sva = np.cross(wa,r2) #surface velocity of active particle svh = np.cross(wh,r1) #surface velocity of home particle vtot =[v1[0]+sva[0]-svh[0],v1[1]+sva[1]-svh[1],v1[2]+sva[2]-svh[2]] #total velocity at the surface of the interaction from the reference of the active particle #print(vtot," Mag ",math.sqrt(vtot[0]**2+vtot[1]**2+vtot[2]**2)) #Since friction coefficient is proportional to velocity difference friction=-visc/(math.sqrt(d[0]**2+d[1]**2+d[2]**2))*Area Fc = [vtot[0]*friction,vtot[1]*friction,vtot[2]*friction] FcM = math.sqrt(Fc[0]**2+Fc[1]**2+Fc[2]**2) if(FcM>Fmax): Fmax=FcM Tq = np.cross(r1,Fc) #torque for home particle I = 2/5*m *(r1[0]**2+r1[1]**2+r1[2]**2) #I=2/5*mr^2 #α = T/I A = [Tq[0]/I,Tq[1]/I,Tq[2]/I] changeInWX[ind] += A[0] changeInWY[ind] += A[1] changeInWZ[ind] += A[2] #since the difference is in the orientation of the radius vector it's as simple as flipping a sign changeInWX[adjacentIndices[i]] -= A[0] changeInWY[adjacentIndices[i]] -= A[1] changeInWZ[adjacentIndices[i]] -= A[2] #Friction effect on the particles in general changeInVX[adjacentIndices[i]] -= Fc[0] changeInVX[ind] += Fc[0] changeInVY[adjacentIndices[i]] -= Fc[1] changeInVY[ind] += Fc[1] changeInVZ[adjacentIndices[i]] -= Fc[2] changeInVZ[ind] += Fc[2] if(particle.t > bondingTemp): #if not within temp requirements (too hot) and previously bonded wipe bonding #essentially the particle has remelted i=0 while(iforceCap): #F=forceCap #elif(F<-forceCap): #F=-forceCap if(F>Fmax): Fmax=F if(not(particleList[ind].isPlate())): #print("index: ",ind, " active: ",adjacentIndices[i]) #print("dist ",dist) #print("F ",F) pass changeInVX[adjacentIndices[i]] -= F/m*(particleActive.x-particle.x)/dist changeInVX[ind] += F/m*(particleActive.x-particle.x)/dist changeInVY[adjacentIndices[i]] -= F/m*(particleActive.y-particle.y)/dist changeInVY[ind] += F/m*(particleActive.y-particle.y)/dist changeInVZ[adjacentIndices[i]] -= F/m*(particleActive.z-particle.z)/dist changeInVZ[ind] += F/m*(particleActive.z-particle.z)/dist #Torque and Angular Momentum d = [(particleActive.x-particle.x)/1000,(particleActive.y-particle.y)/1000,(particleActive.z-particle.z)/1000] #distance vector in mm converting to m v1 = [particleActive.Sx-particle.Sx,particleActive.Sy-particle.Sy,particleActive.Sz-particle.Sz] #bare velocity difference wa =[particleActive.wx,particleActive.wy,particleActive.wz] #angular velocities wh =[particle.wx,particle.wy,particle.wz] r1 = [d[0]/2,d[1]/2,d[2]/2] #'radius' vector of home particle r2 = [-r1[0],-r1[1],-r1[2]] #'radius' vector of active particle sva = np.cross(wa,r2) #surface velocity of active particle svh = np.cross(wh,r1) #surface velocity of home particle vtot =[v1[0]+sva[0]-svh[0],v1[1]+sva[1]-svh[1],v1[2]+sva[2]-svh[2]] #total velocity at the surface of the interaction from the reference of the active particle #print(vtot," Mag ",math.sqrt(vtot[0]**2+vtot[1]**2+vtot[2]**2)) #drag equation + temp change for viscosity TChange = particle.t/bondingTemp*16-15 friction=-visc/(math.sqrt(d[0]**2+d[1]**2+d[2]**2))/TChange*Area Fc = [vtot[0]*friction,vtot[1]*friction,vtot[2]*friction] FcM = math.sqrt(Fc[0]**2+Fc[1]**2+Fc[2]**2) if(FcM>Fmax): Fmax=FcM Tq = np.cross(r1,Fc) #torque for home particle I = 2/5*m *(r1[0]**2+r1[1]**2+r1[2]**2) #I=2/5*mr^2 #α = T/I A= [Tq[0]/I,Tq[1]/I,Tq[2]/I] changeInWX[ind] += A[0] changeInWY[ind] += A[1] changeInWZ[ind] += A[2] #since the difference is in the orientation of the radius vector it's as simple as flipping a sign changeInWX[adjacentIndices[i]] -= A[0] changeInWY[adjacentIndices[i]] -= A[1] changeInWZ[adjacentIndices[i]] -= A[2] #Friction effect on the particles in general changeInVX[adjacentIndices[i]] -= Fc[0] changeInVX[ind] += Fc[0] changeInVY[adjacentIndices[i]] -= Fc[1] changeInVY[ind] += Fc[1] changeInVZ[adjacentIndices[i]] -= Fc[2] changeInVZ[ind] += Fc[2] i+=1 #when a particle passes below a certain temperature it will hold on to its neighbors tightly but will not hold onto others tightly The tightness depends on the length of the bond if(particle.t < bondingTemp and not(particle.isPlate()) and nextEnter != 0): #glass point i=0 while(iVmax): Vmax = speedSq i+=1 Vmax = math.sqrt(Vmax) ''' timeStep = .000001 #timeStep1 = .1*m/Fmax #find timestep such that the change in velocity is 100 mm/s #if(Vmax!=0): #timeStep2 = .0000001/Vmax #find timestep such that max distance moved is .001 mm --> enforces smooth consideration of potential #if(systemTime1>1): #print("V") #else: #timeStep2 = .0005 #if(systemTime1>1): #print("F") #if(timeStep1.0005): #timeStep =.0005 #if(not(particleList[i].isPlate())): #print("F ",Fmax) #print("Time ",systemTimeI) systemTime1 = timeStep + systemTimeI i = 0 while(i< particleNum): #Heat particleList[i].changeTemp(particleList[i].t+changeInHeat[i]*timeStep) #Position if(not(particleList[i].isPlate())): #print("x pos ",particleList[i].Sx*timeStep+particleList[i].x) #print("y pos ",particleList[i].Sy*timeStep+particleList[i].y) #print("z pos ",particleList[i].Sz*timeStep+particleList[i].z) pass particleList[i].changeX(particleList[i].Sx*timeStep*1000+particleList[i].x) #from m/s to mm/s particleList[i].changeY(particleList[i].Sy*timeStep*1000+particleList[i].y) particleList[i].changeZ(particleList[i].Sz*timeStep*1000+particleList[i].z) #Velocity (deliberately calculated after position such that change in velocity is NOT incorporated into that timeStep's motion) speed = particleList[i].Sx+changeInVX[i]*timeStep #if(not(particleList[i].isPlate())): #print(speed) if(speed>speedCap): #Speed Caps to dampen oscillations particleList[i].changeVX(speedCap) elif(speed<-speedCap): particleList[i].changeVX(-speedCap) else: particleList[i].changeVX(particleList[i].Sx+changeInVX[i]*timeStep) speed = particleList[i].Sy+changeInVY[i]*timeStep #if(not(particleList[i].isPlate())): #print(speed) if(speed>speedCap): #Speed Caps to dampen oscillations particleList[i].changeVY(speedCap) elif(speed<-speedCap): particleList[i].changeVY(-speedCap) else: particleList[i].changeVY(particleList[i].Sy+changeInVY[i]*timeStep) speed = particleList[i].Sz+changeInVZ[i]*timeStep #if(not(particleList[i].isPlate())): #print(speed) if(speed>speedCap): #Speed Caps to dampen oscillations particleList[i].changeVZ(speedCap) elif(speed<-speedCap): particleList[i].changeVZ(-speedCap) else: particleList[i].changeVZ(particleList[i].Sz+changeInVZ[i]*timeStep) #Angular Momentum particleList[i].changeWX(particleList[i].wx+changeInWX[i]*timeStep) particleList[i].changeWY(particleList[i].wy*changeInWY[i]*timeStep) particleList[i].changeWZ(particleList[i].wz*changeInWZ[i]*timeStep) #The actual angular 'location' isn't important since the objects are rotationally symmetric i+=1 return systemTime1 def gradeBonds(pList,nextEnter,home,pPotenList): score=0 i=0 while(ilen(particleList)): particleNum = len(particleList) XHist=np.array([None]*particleNum) YHist=np.array([None]*particleNum) ZHist=np.array([None]*particleNum) THist=np.array([None]*particleNum) QHist=np.array([None]*particleNum) i=0 while(ilen(particleList)): particleNum = len(particleList) ''' numPrints = 5 start = 1 #master loop n=0 while(systemTimenumPrints*start-1): saveData(systemTime) start+=1 #print(Fmax) #print(Vmax) length = len(particleList) XHist=np.array([None]*length) YHist=np.array([None]*length) ZHist=np.array([None]*length) THist=np.array([None]*length) QHist=np.array([None]*length) i=0 while(i