diff --git a/.gitignore b/.gitignore index c79736c..f144dbd 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,4 @@ .idea build/ .DS_Store -cmake-build-debug +cmake-build-default/ diff --git a/HistogramGrid.h b/HistogramGrid.h index 3557b3f..c0dfa67 100755 --- a/HistogramGrid.h +++ b/HistogramGrid.h @@ -34,10 +34,15 @@ class HistogramGrid { // int histWidth - Width of the entire histogram in meters // int histLength - Length of the entire histogram in meters // int nodeSideLen - Side dimension of each node. histWidth and histLength should be divisible by this number - HistogramGrid(int histWidth, int histLength, double nodeSideLen): - iMax((int)(histWidth/nodeSideLen)), jMax((int)(histLength/nodeSideLen)), - nodeSize(nodeSideLen), histGrid(new double*[iMax]), objectGrid(new int*[jMax]), - iSizeActiveRegion(10), jSizeActiveRegion(10) + HistogramGrid(int histWidth, int histLength, double nodeSideLen, + int activeRegionSize_i, int activeRegionSize_j): + iMax((int)(histWidth/nodeSideLen)), + jMax((int)(histLength/nodeSideLen)), + nodeSize(nodeSideLen), + histGrid(new double*[iMax]), + objectGrid(new int*[jMax]), + iSizeActiveRegion(activeRegionSize_i), + jSizeActiveRegion(activeRegionSize_j) { std::cout<<"grid: iMax = "<> data; - histWidth = std::stoi(data); - - file >> data; - histLength = std::stoi(data); - - file >> data; - nodeSize = std::stod(data); + // file >> data; + // histWidth = std::stoi(data); + // + // file >> data; + // histLength = std::stoi(data); + // + // file >> data; + // nodeSize = std::stod(data); iMax = (int)(histWidth/nodeSize); jMax = (int)(histLength/nodeSize); histGrid = new double*[iMax]; objectGrid = new int*[jMax]; - iSizeActiveRegion = 20; - jSizeActiveRegion = 20; for(int i = 0; i < iMax; i++) { diff --git a/Plotter.h b/Plotter.h index 412b363..f4b37f5 100644 --- a/Plotter.h +++ b/Plotter.h @@ -7,7 +7,7 @@ #include #include #include "GNUPlot.h" - +#include #ifndef VECTORFIELDHISTOGRAMTESTING_PLOTTER_H #define VECTORFIELDHISTOGRAMTESTING_PLOTTER_H @@ -22,16 +22,20 @@ class Plotter { public: Plotter() { + // TODO: parameterize xrange, yrange according + int xMax = 60; + int yMax = 60; + int pointSize = 7; + std::vector script; f.open(fPath); f << header << std::endl; f.close(); std::this_thread::sleep_for(std::chrono::milliseconds(100)); - - script.push_back("set xrange [0:60];"); - script.push_back("set yrange [0:60];"); - script.push_back("plot \"" + fPath + "\"" + "with points pointsize 7"); + script.push_back("set xrange [0:" + std::to_string(xMax) + "];"); + script.push_back("set yrange [0:" + std::to_string(yMax) + "];"); + script.push_back("plot \"" + fPath + "\"" + "with points pointsize " + std::to_string(pointSize) + ""); plotter.open(); plotter.execute(script); @@ -44,7 +48,7 @@ class Plotter { plotter.close(); } - void plot(std::vector& dataIn) + void plot(std::vector& dataIn, int robotX, int robotY, double angle, int timestep, int numTimesteps) { std::vector script; script.push_back("replot"); @@ -56,6 +60,7 @@ class Plotter { f << dataIn[j].x << " " << dataIn[j].y << std::endl; } f.close(); + script.push_back("set label 11 center at graph 0.2,char 1 \"t = " + std::to_string(timestep) + " / " + std::to_string(numTimesteps) + " position = (" + std::to_string(robotX) + ", " + std::to_string(robotY) + "); angle = " + std::to_string(angle) + "\" font \",14\"; set bmargin 4;"); plotter.execute(script); } }; diff --git a/PolarHistogram.h b/PolarHistogram.h index 68f9a48..abc2c12 100755 --- a/PolarHistogram.h +++ b/PolarHistogram.h @@ -41,6 +41,12 @@ class PolarHistogram { histogram[9] = 3; } + //getIndex + //Returns index 0 <= i < nBins that corresponds to a. "Wraps" a around histogram. + int getIndex(int a) { + return ((a % nBins) + nBins) % nBins; + } + //getBinFromAngle //Returns the index of the bin based on the angle relative to the absolute coordinate system with which //histogram is represented @@ -53,7 +59,7 @@ class PolarHistogram { //Returns the angle in the middle of the bin double getAngleFromBin(int bin) { - return binWidth/2 + binWidth*bin; + return binWidth/2 + binWidth*getIndex(bin); } //getNumBins @@ -67,7 +73,8 @@ class PolarHistogram { //Returns the value of the histogram for the specified bin double getValue(int bin) { - return histogram[bin]; + + return histogram[getIndex(bin)]; } //addValue @@ -101,7 +108,7 @@ class PolarHistogram { { for(int i = 0; i < nBins; i++) { - std::cout << getAngleFromBin(i) << " " << histogram[i] << "\n"; + std::cout << i <<"th bin: angle = " << getAngleFromBin(i) << " , certainty = " << histogram[i] << "\n"; } } diff --git a/README.md b/README.md index 82057f1..fad6faa 100755 --- a/README.md +++ b/README.md @@ -1 +1,6 @@ -"# VectorFieldHistogramTesting" +# VectorFieldHistogramTesting + +Bugs + +- [] contpoint vs discretepoint angle issue. Error may be due to discretization. +- [] smoothing? diff --git a/RobotTest.h b/RobotTest.h index cc1cf1d..d704b6e 100644 --- a/RobotTest.h +++ b/RobotTest.h @@ -1,6 +1,7 @@ -// -// Created by Phil on 2/3/2018. -// +// RobotTest.h +// @author {Zhanwen Chen, Swapnil Pande} +// @date Feb 3, 2018 + #ifndef VECTORFIELDHISTOGRAMTESTING_ROBOTTEST_CPP #define VECTORFIELDHISTOGRAMTESTING_ROBOTTEST_CPP @@ -8,143 +9,119 @@ #include #include "VFHPather.h" #include "Utils.h" -// #include "gnuplot-iostream.h" #include "Plotter.h" - +#define HIST_WIDTH 50 // originally 50 +#define HIST_LENGTH 50 // originally 50 +#define NODE_SIZE_IN 1 +#define NUM_BINS 32 // 32 originally. 25 gets around to the target +#define ACTIVE_REGION_SIZE_I 30 // Originally 20. 30 gets stuck early on. 10 Goes through all obstacles. +#define ACTIVE_REGION_SIZE_J 30 + +// Less useful params +#define TARGET_X 50 +#define TARGET_Y 50 +#define A_IN 200 +#define B_IN 1 +#define L_IN 5 +#define MAX_NUM_NODES_FOR_VALLEY_IN 15 +#define VALLEY_THRESHOLD_IN 10 + +// #define MAP_FNAME "../map.txt" + +/* +* NOTE: it is important to distinguish between the same variables at +* t versus t-1. Instance variables are shared across two timesteps. +*/ class RobotTest{ private: HistogramGrid grid; PolarHistogram hist; VFHPather pather; //Object used to store the grid/map of obstacles - discretePoint currentPosition; //Stores the location of the robot - double currentSpeed; // linear distance per timestep - double maxTurnSpeed; // angle per timestep - double currentAbsoluteAngle; //absolute + discretePoint location; //Stores the location of the robot + double speed; // linear distance per timestep + // double maxTurnSpeed; // angle per timestep REVIEW: necessary? + double angle; // In degrees. Plotter plotter; - // double[][] world -public: - //RobotTest - //Constructor for the RobotTest class - // init_x: (int) - // init_y: (int) - RobotTest(discretePoint initPos, double angle_init, double speed_init): grid("../map.txt", initPos), hist(32), - pather(hist, &grid, 100, 1, 5, 15), - currentPosition(initPos), - currentSpeed(speed_init), - currentAbsoluteAngle(angle_init) - // pather(20, 50, 100, 0.1) // iMax = 10, jMax = 10?? - // iMax = 10, jMax = 10?? + void draw(int currentTimestep, int numTimesteps) { - pather.updateRobotPosition(currentPosition); - grid.setTargetLoc({50,50}); + int iMax = pather.getIMax(); + int jMax = pather.getJMax(); + std::vector positions; + positions.push_back(grid.getRobotLoc()); + positions.push_back(grid.getTargetLoc()); + for(int i = 0; i < iMax; ++i) + { + for(int j = 0; j < jMax; ++j) + { + if(pather.getCellValue(i, j) == 1) + { + positions.push_back({i, j}); + } + } + } + plotter.plot(positions, location.x, location.y, angle, currentTimestep, numTimesteps); } -// ~RobotTest() -// { -// // delete pather; // FIXME: no delete method in them -// // delete currentPosition; // FIXME: no delete method in them -// } - - //main function per timestep - void move() + void getAngle() { - updateRobotAngle(); - updateRobotSpeed(); - updateRobotPosition(); + angle = pather.computeTravelDirection(); + std::cout << "Desired travel angle: " << angle << "\n"; } - void updateRobotPosition() + // TODO: currently speed is always 1 width per timestep + void getSpeed() { - currentPosition.x += currentSpeed * cos(currentAbsoluteAngle*M_PI/180); - currentPosition.y += currentSpeed * sin(currentAbsoluteAngle*M_PI/180); - // std::cout<<"trying to update the robot position.\n"; - pather.updateRobotPosition(currentPosition); - // pather.generateHistogram(); + speed = 2; } - void updateRobotAngle() + void updateLocation() { - // currentAbsoluteAngle = pather.getMinHistAngleBetweenAngles(currentPosition, currentAbsoluteAngle, maxTurnSpeed); - // std::cout<<"trying to update the robot angle.\n"; - currentAbsoluteAngle = pather.computeTravelDirection(); - std::cout << "Desired travel angle: " << currentAbsoluteAngle << "\n"; - //hist.printHistogram(); + location.x += speed * cos(angle*M_PI/180); + location.y += speed * sin(angle*M_PI/180); + pather.updateRobotPosition(location); } - // TODO: currently speed is always 1 width per timestep - void updateRobotSpeed() + + void talk() { - currentSpeed = 2; + discretePoint targetLoc = grid.getTargetLoc(); + std::cout << "location is (" << location.x << ", " + << location.y << "); Desired Position is (" + << targetLoc.x << ", " << targetLoc.y << ")\n"; } - void talk() +public: + //RobotTest + //Constructor for the RobotTest class + //CHANGED: no need for initAngle or initSpeed: robot should figure out. + // RobotTest(discretePoint initPos, double initAngle, double initSpeed): + RobotTest(discretePoint initPos): + grid("../map.txt", initPos, ACTIVE_REGION_SIZE_I, ACTIVE_REGION_SIZE_J, HIST_WIDTH, HIST_LENGTH, NODE_SIZE_IN), + hist(NUM_BINS), + pather(hist, &grid, A_IN, B_IN, L_IN, MAX_NUM_NODES_FOR_VALLEY_IN, VALLEY_THRESHOLD_IN), + location(initPos) { - std::cout<<"currentPosition is "< positions; - // std::iota(std::begin(x), std::end(x), 0); //0 is the starting number - positions.push_back(grid.getRobotLoc()); - for(int i = 0; i < iMax; ++i) - { - for(int j = 0; j < jMax; ++j) - { - //std::cout << pather.getCellValue(i, j) << ' '; - if(pather.getCellValue(i, j) == 1) - { - //positions.push_back(grid.getRobotLoc()); - positions.push_back({i, j}); - //std::cout << "pushing obstacle at (" << i << ", " << j << ")\n"; - } - } + getAngle(); // angle: Null (or optionally, t-1) => t + getSpeed(); // speed: Null (or optionally, t-1) => t - } - plotter.plot(positions); - // try { - // - // // Don't forget to put "\n" at the end of each line! - // // gp << "set xrange [-2:2]\nset yrange [-2:2]\n"; - // // '-' means read from stdin. The send1d() function sends data to gnuplot's stdin. - // // gp << "plot '.' with vectors title 'x', '-' with vectors title 'y'\n"; - // - // // plot.set_style("points").set_xrange(0, iMax).set_yrange(0, jMax).plot_xy(x, y, "points"); - // - // // plot.reset_plot(); - // } - // - // catch (GnuplotException ge) - // { - // std::cout << ge.what() << std::endl; - // } - - - // g2.plot_xy(); - // int** objectGrid = pather.getObjectGrid(); - // objectGrid[currentPosition.x][currentPosition.y] = 1; - - // int nrows = sizeof objectGrid / sizeof objectGrid[0]; - - // int ncols = sizeof objectGrid[0] / sizeof(int); - - // std::cout<<"objectGrid: nrows = "<getTargetLoc())); //Determine the bin in which the target falls + int startBin = polarHist.getBinFromAngle((*grid).getAngle((*grid).getRobotLoc(), grid->getTargetLoc())); //Determine the bin in which the target falls - int negative = 1; //Used to alternate the direction of the array iteration - int nearIndex = -1; //Index of the edge of valley closest to the target - int farIndex = -1; //Index of the edge of the valley furthest from target - //Determining if the target direction falls within a bin - if(hist.getValue(startBin) < valleyThreshold) //Desired travel direction falls within a valley + //EDGE CASE: Determining if the target direction falls within a bin + //This handled by finding the edges of the valleys + if(polarHist.getValue(startBin) < valleyThreshold) //Desired travel direction falls within a valley { - std::cout << "TEST 1 --------------\n"; - //Find upper boundary of valley - for(int i = startBin + 1; getIndex(i, hist.getNumBins()) != startBin; i++) + std::cout << "\n\n\nREACHED EDGE CASE\n\n\n"; + int leftIndex = -1, rightIndex = -1; //Store the indices of the edges of the valley + + int negative = 1; //Used to the flip the direction of search + int divide = 2; //Used to control the direction search occurs + + int count = 1; //Counter to store number of bins tested + int i; //Stores index to search + //Iterating over the histogram to find valley. + //Iteration occurs alternating in left & right direction of target + while(count <= polarHist.getNumBins() && count <= maxNumNodesForValley) { - //std::cout << getIndex(i, hist.getNumBins()) << " "; - if(hist.getValue(getIndex(i, hist.getNumBins())) > valleyThreshold) + + i = startBin + negative * count / divide; //Index of bin to check next + + if(polarHist.getValue(i) > valleyThreshold) { - farIndex = i; //Found the further edge of the valley - break; - } - } - std::cout << "\n"; - std::cout << "TEST 2 --------------\n"; + if(negative == 1) rightIndex = i + 1; + else leftIndex = i; - //Was not able to find the edge of a valley. Therefore, there are no obstacles. - //Robot should travel straight to target. - if(farIndex == -1) - { - return hist.getAngleFromBin(startBin); - } - std::cout << "TEST 7 --------------\n"; + //One bound of index is found. Only need to search in other direction + negative *= -1; + divide = 1; + } + //Flipping search direction if both ends of the valley have not been found + if(leftIndex == -1 && rightIndex == -1) negative *= -1; - //Find lower boundary of valley - for(int i = startBin - 1; getIndex(i, hist.getNumBins()) != startBin; i--) + count++; //Increment counter + } + if(count > maxNumNodesForValley) //The maximum size of a valley was reached. Write the edges at the last searched bins { - //std::cout << hist.getValue(hist.getValue(getIndex(i, hist.getNumBins()))) << "\n\n\n"; - if(hist.getValue(getIndex(i, hist.getNumBins())) > valleyThreshold) + //Stores the edges of the valley if the size has reached maxNumNodesForValley. + //If divide is 2, neither edge of the boundary has been found. + //If divide is 1, one edge of the boundary has been found. + for(int j = 0; j < divide; j++) { - nearIndex = i + 1; //Found the nearer edge of the valley - break; + count--; + i = startBin + negative * count / divide; //Index of bin to check next + + if(negative == 1) rightIndex = i + 1; //Storing edge of valley + else leftIndex = i; } } - std::cout << "TEST 3 --------------\n"; - std::this_thread::sleep_for(std::chrono::milliseconds(500)); - - return hist.getAngleFromBin(getIndex((farIndex+nearIndex)/2, hist.getNumBins())); + // std::cout << leftIndex << " " << rightIndex << "\n"; + //Returns the average value of the bins + return polarHist.getAngleFromBin((rightIndex+leftIndex)/2); } - else + else //Not edge case, process histogram normally { - //Finding other valleys in histogram - //Looping over the entire polar histogram - //Count stores the number of bins that have been checked - //Loop begins iterating at the startBin, and alternates back and forth to find nearest valley - for (int count = 1; count <= hist.getNumBins(); count = count + 1) { - int i = getIndex(startBin + negative * count / 2, hist.getNumBins()); //Index of bin to check next - if (hist.getValue(i) < valleyThreshold) //Found valley + int rightTravelDir = -1, leftTravelDir = -1; //Travel direction for valleys nearest to target on left and right of target + //Finding nearest left and right valleys + //Looping over histogram to left of target, then to the right of target + //Stores the suggested travel direction for each of side of the histogram and selects direction closest to target + int i; + //Checking left side + // std::cout << "\n Left Side parameters: \n"; + for (i = startBin + 1; i <= polarHist.getNumBins()/2; i++) { + if (polarHist.getValue(i) < valleyThreshold) //Found valley { - nearIndex = i; - count++; - i = getIndex(i + negative, hist.getNumBins()); //Adds or subtracts index based on direction of array traversal. - while (count <= hist.getNumBins() && hist.getValue(i) < valleyThreshold) { - i = getIndex(i + negative, hist.getNumBins()); - } - farIndex = i; + int rightIndex = i; + //Iterating over valley to find other edge of the valley. + while(polarHist.getValue(i) < valleyThreshold && abs(i - rightIndex) < maxNumNodesForValley) i++; + int leftIndex = i; + leftTravelDir = (rightIndex + leftIndex)/2; + // std::cout <<"lol"<< polarHist.getIndex(leftIndex) << " " << polarHist.getIndex(rightIndex) << " " << polarHist.getIndex(leftTravelDir) << "\n"; break; //Since loop begins iterating from the target direction, the valley must be the closest valley } - negative *= -1; //Flipping direction } - } - //Edges of nearest valley are determined, must determine travel direction - int travelDirectionIndex; - if(startBin > nearIndex && farIndex > nearIndex) - { - travelDirectionIndex = getIndex((nearIndex + farIndex + hist.getNumBins())/2, hist.getNumBins()); - } - else if(startBin < nearIndex && farIndex < nearIndex) - { - travelDirectionIndex = getIndex((nearIndex - hist.getNumBins() + farIndex )/2, hist.getNumBins()); - } - else - { - travelDirectionIndex = (nearIndex + farIndex)/2; + if(i < polarHist.getNumBins()/2) i = polarHist.getNumBins() - polarHist.getNumBins()/2; //setting max iteration for left side + i -= polarHist.getNumBins(); + // std::cout << "\ni: " << i << " " << polarHist.getIndex(i) << std::endl; + int j; + //Checking right side + // std::cout << "\n Right Side parameters: \n"; + for (j = startBin - 1; j > i; j--) { + // std::cout << "\nj: " << j << " " << i << " " << polarHist.getValue(j) << "\n"; + if (polarHist.getValue(j) < valleyThreshold) //Found valley + { + int rightIndex = j+1; + //Iterating over valley to find other edge of the valley. + while(polarHist.getValue(j) < valleyThreshold && abs(j-rightIndex) < maxNumNodesForValley) + { + j--; + } + int leftIndex = j+1; + leftTravelDir = (rightIndex + leftIndex)/2; + // std::cout << "lol"<> data(2, std::vector(0, 0)); -// data[0].push_back(0); -// data[1].push_back(0); -// for(int i = 0; i < 500; i++) -// { -// data[0][0] = i; -// data[1][0] = i; -// -// std::cout << i << std::endl; -// plotter.plot(data); -// } - - discretePoint init; - init.x = 30; - init.y = 30; + init.x = 0; + init.y = 0; // double angle_init = 0.78; // pi/4 - double angle_init = 90; // pi/4 - double speed_init = 0.5; - - RobotTest bot(init, angle_init, speed_init); + // double angle_init = 90; // pi/4 + // double speed_init = 0.5; - int timestep = 100; + RobotTest bot(init); + getchar(); + int numTimesteps = 100; - for(int i = 0; i < timestep; ++i) + for(int i = 0; i < numTimesteps; ++i) { - std::cout<