// // 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); } }