228 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			228 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //
 | |
| //  DLASystem.cpp
 | |
| //
 | |
| 
 | |
| #include "DLASystem.h"
 | |
| 
 | |
| using std::cout;
 | |
| using std::endl;
 | |
| 
 | |
| // this function gets called every step,
 | |
| //   if there is an active particle then it gets moved,
 | |
| //   if not then add a particle
 | |
| void DLASystem::Update() {
 | |
|     this->frame++;
 | |
| 
 | |
|     if (lastParticleIsActive == 1) {
 | |
|         moveLastParticle();
 | |
|     } else if (this->particleList.size() < (size_t) endNum) {
 | |
|         addParticleOnAddCircle();
 | |
|         setParticleActive();
 | |
|     } else {
 | |
|         this->running = false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void DLASystem::clearParticles() {
 | |
|     // delete particles and the particle list
 | |
|     for (size_t i = 0; i < this->particleList.size(); i++) {
 | |
|         delete particleList[i];
 | |
|     }
 | |
|     particleList.clear();
 | |
| }
 | |
| 
 | |
| // remove any existing particles and setup initial condition
 | |
| void DLASystem::Reset() {
 | |
|     // stop running
 | |
|     running = false;
 | |
|     frame = 0;
 | |
| 
 | |
|     clearParticles();
 | |
| 
 | |
|     lastParticleIsActive = 0;
 | |
| 
 | |
|     // set the grid to zero
 | |
|     for (int i = 0; i < gridSize; i++) {
 | |
|         for (int j = 0; j < gridSize; j++) {
 | |
|             grid[i][j] = 0;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // setup initial condition and parameters
 | |
|     addCircle = 10;
 | |
|     killCircle = 2.0 * addCircle;
 | |
|     clusterRadius = 0.0;
 | |
|     // add a single particle at the origin
 | |
|     double pos[] = {0.0, 0.0};
 | |
|     addParticle(pos);
 | |
| }
 | |
| 
 | |
| // set the value of a grid cell for a particular position
 | |
| // note the position has the initial particle at (0,0)
 | |
| // but this corresponds to the middle of the grid array ie grid[ halfGrid ][ halfGrid ]
 | |
| void DLASystem::setGrid(double pos[], int val) {
 | |
|     int halfGrid = gridSize / 2;
 | |
|     grid[(int) (pos[0] + halfGrid)][(int) (pos[1] + halfGrid)] = val;
 | |
| }
 | |
| 
 | |
| // read the grid cell for a given position
 | |
| int DLASystem::readGrid(double pos[]) {
 | |
|     int halfGrid = gridSize / 2;
 | |
|     return grid[(int) (pos[0] + halfGrid)][(int) (pos[1] + halfGrid)];
 | |
| }
 | |
| 
 | |
| // add a particle to the system at a specific position
 | |
| void DLASystem::addParticle(double pos[]) {
 | |
|     // create a new particle
 | |
|     Particle *p = new Particle(pos);
 | |
|     // push_back means "add this to the end of the list"
 | |
|     particleList.push_back(p);
 | |
| 
 | |
|     // pos coordinates should be -gridSize/2 < x < gridSize/2
 | |
|     setGrid(pos, 1);
 | |
| }
 | |
| 
 | |
| // add a particle to the system at a random position on the addCircle
 | |
| // if we hit an occupied site then we do nothing except print a message
 | |
| // (this should never happen)
 | |
| void DLASystem::addParticleOnAddCircle() {
 | |
|     double pos[2];
 | |
|     double theta = rgen.random01() * 2 * M_PI;
 | |
|     pos[0] = ceil(addCircle * cos(theta));
 | |
|     pos[1] = ceil(addCircle * sin(theta));
 | |
|     if (readGrid(pos) == 0) {
 | |
|         addParticle(pos);
 | |
|     } else {
 | |
|         cout << "FAIL " << pos[0] << " " << pos[1] << endl;
 | |
|     }
 | |
| }
 | |
| 
 | |
| // send back the position of a neighbour of a given grid cell
 | |
| // NOTE: there is no check that the neighbour is inside the grid,
 | |
| // this has to be done separately...
 | |
| void DLASystem::setPosNeighbour(double setpos[], double pos[], int val) {
 | |
|     switch (val) {
 | |
|         case 0:
 | |
|             setpos[0] = pos[0] + 1.0;
 | |
|             setpos[1] = pos[1];
 | |
|             break;
 | |
|         case 1:
 | |
|             setpos[0] = pos[0] - 1.0;
 | |
|             setpos[1] = pos[1];
 | |
|             break;
 | |
|         case 2:
 | |
|             setpos[0] = pos[0];
 | |
|             setpos[1] = pos[1] + 1.0;
 | |
|             break;
 | |
|         case 3:
 | |
|             setpos[0] = pos[0];
 | |
|             setpos[1] = pos[1] - 1.0;
 | |
|             break;
 | |
|     }
 | |
| }
 | |
| 
 | |
| // when we add a particle to the cluster, we should update the cluster radius
 | |
| // and the sizes of the addCircle and the killCircle
 | |
| void DLASystem::updateClusterRadius(double pos[]) {
 | |
|     double newRadius = distanceFromOrigin(pos);
 | |
|     if (newRadius > clusterRadius) {
 | |
|         clusterRadius = newRadius;
 | |
|         // this is how big addCircle is supposed to be:
 | |
|         //   either 20% more than cluster radius, or at least 5 bigger.
 | |
|         double check = clusterRadius * addRatio;
 | |
|         if (check < clusterRadius + 5) {
 | |
|             check = clusterRadius + 5;
 | |
|         }
 | |
| 
 | |
|         // if it is smaller then update everything...
 | |
|         if (addCircle < check) {
 | |
|             addCircle = check;
 | |
|             killCircle = killRatio * addCircle;
 | |
|         }
 | |
| 
 | |
|         if (killCircle + 2 >= gridSize / 2) {
 | |
|             this->running = false;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // make a random move of the last particle in the particleList
 | |
| void DLASystem::moveLastParticle() {
 | |
|     int direction = rgen.randomInt(4);  // pick a random number in the range 0-3, which direction do we hop?
 | |
|     double newpos[2];
 | |
| 
 | |
|     Particle *lastP = particleList[this->particleList.size() - 1];
 | |
| 
 | |
|     setPosNeighbour(newpos, lastP->pos, direction);
 | |
| 
 | |
|     if (distanceFromOrigin(newpos) > killCircle) {
 | |
|         setGrid(lastP->pos, 0);
 | |
|         particleList.pop_back();  // remove particle from particleList
 | |
|         setParticleInactive();
 | |
|     } else if (readGrid(newpos) == 0) {
 | |
|         setGrid(lastP->pos, 0);  // set the old grid site to empty
 | |
|         // update the position
 | |
|         particleList[this->particleList.size() - 1]->pos[0] = newpos[0];
 | |
|         particleList[this->particleList.size() - 1]->pos[1] = newpos[1];
 | |
|         setGrid(lastP->pos, 1);  // set the new grid site to be occupied
 | |
| 
 | |
|         // check if we stick
 | |
|         if (checkStick() && this->rgen.random01() <= this->stickProbability) {
 | |
|             this->csv_out << frame << "," << lastP->pos[0] << "," << lastP->pos[1] << endl;
 | |
|             setParticleInactive();  // make the particle inactive (stuck)
 | |
|             updateClusterRadius(lastP->pos);  // update the cluster radius, addCircle, etc.
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // check if the last particle should stick (to a neighbour)
 | |
| int DLASystem::checkStick() {
 | |
|     Particle *lastP = particleList[this->particleList.size() - 1];
 | |
|     int result = 0;
 | |
|     // loop over neighbours
 | |
|     for (int i = 0; i < 4; i++) {
 | |
|         double checkpos[2];
 | |
|         setPosNeighbour(checkpos, lastP->pos, i);
 | |
|         // if the neighbour is occupied...
 | |
|         if (readGrid(checkpos) == 1)
 | |
|             result = 1;
 | |
|     }
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| 
 | |
| // constructor
 | |
| DLASystem::DLASystem(Config config)
 | |
|         : stickProbability(config.stickProbability), csv_out(std::move(config.csv)), endNum(config.maxParticles) {
 | |
|     cout << "creating system, gridSize " << gridSize << endl;
 | |
|     rgen.setSeed(config.seed);
 | |
| 
 | |
|     // allocate memory for the grid, remember to free the memory in destructor
 | |
|     grid = new int *[gridSize];
 | |
|     for (int i = 0; i < gridSize; i++) {
 | |
|         grid[i] = new int[gridSize];
 | |
|     }
 | |
| 
 | |
|     // reset initial parameters
 | |
|     Reset();
 | |
| 
 | |
|     this->csv_out << 0 << "," << 0 << "," << 0 << std::endl;
 | |
| 
 | |
|     addRatio = 1.2;   // how much bigger the addCircle should be, compared to cluster radius
 | |
|     killRatio = 1.7;   // how much bigger is the killCircle, compared to the addCircle
 | |
| }
 | |
| 
 | |
| // destructor
 | |
| DLASystem::~DLASystem() {
 | |
|     // delete the particles
 | |
|     clearParticles();
 | |
|     // delete the grid
 | |
|     for (int i = 0; i < gridSize; i++)
 | |
|         delete[] grid[i];
 | |
|     delete[] grid;
 | |
| 
 | |
|     if (csv_out.is_open()) {
 | |
|         csv_out.close();
 | |
|     }
 | |
| }
 |