320 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			320 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //
 | |
| //  DLASystem.cpp
 | |
| //
 | |
| 
 | |
| #include "DLASystem.h"
 | |
| 
 | |
| // colors
 | |
| namespace colours {
 | |
| 	GLfloat blue[] = { 0.1, 0.3, 0.9, 1.0 };   // blue
 | |
| 	GLfloat red[] = { 1.0, 0.2, 0.1, 0.2 };   // red
 | |
| 	GLfloat green[] = { 0.3, 0.6, 0.3, 1.0 };     // green
 | |
| 	GLfloat paleGrey[] = { 0.7, 0.7, 0.7, 1.0 };     // green
 | |
| 	GLfloat darkGrey[] = { 0.2, 0.2, 0.2, 1.0 };     // green
 | |
| }
 | |
| 
 | |
| 
 | |
| // 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() {
 | |
| 	if (lastParticleIsActive == 1)
 | |
| 		moveLastParticle();
 | |
| 	else if (numParticles < endNum) {
 | |
| 		addParticleOnAddCircle();
 | |
| 		setParticleActive();
 | |
| 	}
 | |
| 	if (lastParticleIsActive == 0 || slowNotFast == 1)
 | |
| 		glutPostRedisplay(); //Tell GLUT that the display has changed
 | |
| }
 | |
| 
 | |
| 
 | |
| void DLASystem::clearParticles() {
 | |
| 	// delete particles and the particle list
 | |
| 	for (int i = 0; i < numParticles; i++) {
 | |
| 		delete particleList[i];
 | |
| 	}
 | |
| 	particleList.clear();
 | |
| 	numParticles = 0;
 | |
| }
 | |
| 
 | |
| // remove any existing particles and setup initial condition
 | |
| void DLASystem::Reset() {
 | |
| 	// stop running
 | |
| 	running = 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 view
 | |
| 	int InitialViewSize = 40;
 | |
| 	setViewSize(InitialViewSize);
 | |
| 
 | |
| }
 | |
| 
 | |
| // 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)];
 | |
| }
 | |
| 
 | |
| // check if the cluster is big enough and we should stop:
 | |
| // to be safe, we need the killCircle to be at least 2 less than the edge of the grid
 | |
| int DLASystem::checkStop() {
 | |
| 	if (killCircle + 2 >= gridSize / 2) {
 | |
| 		pauseRunning();
 | |
| 		cout << "STOP" << endl;
 | |
| 		glutPostRedisplay(); // update display
 | |
| 		return 1;
 | |
| 	}
 | |
| 	else return 0;
 | |
| }
 | |
| 
 | |
| // 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);
 | |
| 	numParticles++;
 | |
| 
 | |
| 	// 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;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // if the view is smaller than the kill circle then increase the view area (zoom out)
 | |
| void DLASystem::updateViewSize() {
 | |
| 	double mult = 1.2;
 | |
| 	if (viewSize < 2.0*killCircle) {
 | |
| 		setViewSize(viewSize * mult);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // set the view to be the size of the add circle (ie zoom in on the cluster)
 | |
| void DLASystem::viewAddCircle() {
 | |
| 	setViewSize(2.0*addCircle);  // factor of 2 is to go from radius to diameter
 | |
| }
 | |
| 
 | |
| // 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 rr = distanceFromOrigin(pos);
 | |
| 	if (rr > clusterRadius) {
 | |
| 		clusterRadius = rr;
 | |
| 		// 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;
 | |
| 			updateViewSize();
 | |
| 		}
 | |
| 		checkStop();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // make a random move of the last particle in the particleList
 | |
| void DLASystem::moveLastParticle() {
 | |
| 	int rr = rgen.randomInt(4);  // pick a random number in the range 0-3, which direction do we hop?
 | |
| 	double newpos[2];
 | |
| 
 | |
| 	Particle *lastP = particleList[numParticles - 1];
 | |
| 
 | |
| 	setPosNeighbour(newpos, lastP->pos, rr);
 | |
| 
 | |
| 	if (distanceFromOrigin(newpos) > killCircle) {
 | |
| 		//cout << "#deleting particle" << endl;
 | |
| 		setGrid(lastP->pos, 0);
 | |
| 		particleList.pop_back();  // remove particle from particleList
 | |
| 		numParticles--;
 | |
| 		setParticleInactive();
 | |
| 	}
 | |
| 	// check if destination is empty
 | |
| 	else if (readGrid(newpos) == 0) {
 | |
| 		setGrid(lastP->pos, 0);  // set the old grid site to empty
 | |
| 		// update the position
 | |
| 		particleList[numParticles - 1]->pos[0] = newpos[0];
 | |
| 		particleList[numParticles - 1]->pos[1] = newpos[1];
 | |
| 		setGrid(lastP->pos, 1);  // set the new grid site to be occupied
 | |
| 
 | |
| 		// check if we stick
 | |
| 		if (checkStick()) {
 | |
| 			//cout << "stick" << endl;
 | |
| 			setParticleInactive();  // make the particle inactive (stuck)
 | |
| 			updateClusterRadius(lastP->pos);  // update the cluster radius, addCircle, etc.
 | |
| 
 | |
| 			if (numParticles % 100 == 0 && logfile.is_open()) {
 | |
| 				logfile << numParticles << " " << clusterRadius << endl;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	else {
 | |
| 		// if we get to here then we are trying to move to an occupied site
 | |
| 		// (this should never happen as long as the sticking probability is 1.0)
 | |
| 		cout << "reject " << rr << endl;
 | |
| 		cout << lastP->pos[0] << " " << lastP->pos[1] << endl;
 | |
| 		//cout << newpos[0] << " " << newpos[1] << " " << (int)newpos[0] << endl;
 | |
| 		//printOccupied();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // check if the last particle should stick (to a neighbour)
 | |
| int DLASystem::checkStick() {
 | |
| 	Particle *lastP = particleList[numParticles - 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(Window *set_win) {
 | |
| 	cout << "creating system, gridSize " << gridSize << endl;
 | |
| 	win = set_win;
 | |
| 	numParticles = 0;
 | |
| 	endNum = 1000;
 | |
| 
 | |
| 	// 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];
 | |
| 	}
 | |
| 	slowNotFast = 1;
 | |
| 	// reset initial parameters
 | |
| 	Reset();
 | |
| 
 | |
| 	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
 | |
| 
 | |
| 	// this opens a logfile, if we want to...
 | |
| 	//logfile.open("opfile.txt");
 | |
| }
 | |
| 
 | |
| // destructor
 | |
| DLASystem::~DLASystem() {
 | |
| 	// strictly we should not print inside the destructor but never mind...
 | |
| 	cout << "deleting system" << endl;
 | |
| 	// delete the particles
 | |
| 	clearParticles();
 | |
| 	// delete the grid
 | |
| 	for (int i = 0; i < gridSize; i++)
 | |
| 		delete[] grid[i];
 | |
| 	delete[] grid;
 | |
| 
 | |
| 	if (logfile.is_open())
 | |
| 		logfile.close();
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| // this draws the system
 | |
| void DLASystem::DrawSquares() {
 | |
| 
 | |
| 	// draw the particles
 | |
| 	double halfSize = 0.5;
 | |
| 	for (int p = 0; p < numParticles; p++) {
 | |
| 		double *vec = particleList[p]->pos;
 | |
| 		glPushMatrix();
 | |
| 		if (p == numParticles - 1 && lastParticleIsActive == 1)
 | |
| 			glColor4fv(colours::red);
 | |
| 		else if (p == 0)
 | |
| 			glColor4fv(colours::green);
 | |
| 		else
 | |
| 			glColor4fv(colours::blue);
 | |
| 		glRectd(drawScale*(vec[0] - halfSize),
 | |
| 			drawScale*(vec[1] - halfSize),
 | |
| 			drawScale*(vec[0] + halfSize),
 | |
| 			drawScale*(vec[1] + halfSize));
 | |
| 		glPopMatrix();
 | |
| 	}
 | |
| 
 | |
| 	// print some information (at top left)
 | |
| 	// this ostringstream is a way to create a string with numbers and words (similar to cout << ... )
 | |
| 	ostringstream str;
 | |
| 	str << "num " << numParticles << " size " << clusterRadius;
 | |
| 
 | |
| 	// print the string
 | |
| 	win->displayString(str, -0.9, 0.9, colours::red);
 | |
| 
 | |
| 	// if we are paused then print this (at bottom left)
 | |
| 	if (running == 0) {
 | |
| 		ostringstream pauseStr;
 | |
| 		pauseStr << "paused";
 | |
| 		win->displayString(pauseStr, -0.9, -0.9, colours::red);
 | |
| 	}
 | |
| 
 | |
| }
 |