From aca9a372b4b97f6f431b41baf6a50a8d7902ee89 Mon Sep 17 00:00:00 2001 From: Swapnil Pande Date: Sat, 3 Mar 2018 17:19:59 -0600 Subject: [PATCH 1/5] V2 valley detection implemented and working. --- PolarHistogram.h | 13 +++- RobotTest.h | 3 +- VFHPather.h | 164 +++++++++++++++++++++++++---------------------- main.cpp | 2 + map.txt | 20 +++--- 5 files changed, 113 insertions(+), 89 deletions(-) diff --git a/PolarHistogram.h b/PolarHistogram.h index 68f9a48..2a37e6a 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 << " " << getAngleFromBin(i) << " " << histogram[i] << "\n"; } } diff --git a/RobotTest.h b/RobotTest.h index cc1cf1d..5ac4389 100644 --- a/RobotTest.h +++ b/RobotTest.h @@ -31,7 +31,7 @@ class RobotTest{ // 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), + pather(hist, &grid, 200, 1, 5, 15), currentPosition(initPos), currentSpeed(speed_init), currentAbsoluteAngle(angle_init) @@ -94,6 +94,7 @@ class RobotTest{ std::vector positions; // std::iota(std::begin(x), std::end(x), 0); //0 is the starting number positions.push_back(grid.getRobotLoc()); + positions.push_back(grid.getTargetLoc()); for(int i = 0; i < iMax; ++i) { for(int j = 0; j < jMax; ++j) diff --git a/VFHPather.h b/VFHPather.h index ef8aefb..72e6602 100755 --- a/VFHPather.h +++ b/VFHPather.h @@ -45,7 +45,7 @@ class VFHPather{ // If polar object density falls below valley threshold, bin is considered to be part of valley VFHPather(PolarHistogram &histIn, HistogramGrid *gridIn, double aIn, double bIn, double lIn, double valleyThresholdIn): hist(histIn), grid(gridIn), a(aIn), b(bIn), l(lIn), - smax(5), valleyThreshold(valleyThresholdIn) {} + smax(10), valleyThreshold(valleyThresholdIn) {} //TODO: Add ability to dynamically set certainty value //TODO This function may be deprecated as we restructure the robot code for ROSMOD //updateRobotPosition @@ -67,10 +67,6 @@ class VFHPather{ discretePoint curNode; //Node currently being iterated over - // std::cout << "Active Region: " << activeRegion.min.x << " " - // << activeRegion.min.y << " " << activeRegion.max.x << " " << activeRegion.max.y << "\n"; - //std::cout << "Histogram generation: \n"; - for(curNode.x = activeRegion.min.x; curNode.x < activeRegion.max.x; curNode.x++) { for(curNode.y = activeRegion.min.y; curNode.y < activeRegion.max.y; curNode.y++) @@ -89,110 +85,128 @@ class VFHPather{ } - //getIndex - //Returns index 0 <= i < n that corresponds to a. "Wraps" a. - int getIndex(int a, int n) { - return ((a % n) + n) % n; - } + //computeTravelDirection //Determines the optimal travel direction for the robot based on the generated polar histogram //Works by finding the valley whose direction most closely matches the direction of the target double computeTravelDirection() { - std::cout << "TEST 0 --------------\n"; generateHistogram(); //Computes the polar histogram printHistogram(); - //hist.smoothHistogram(l); //Smoothing histogram - std::cout << "TEST 0.5 --------------\n"; + //hist.smoothHistogram(l); //Smoothing histogram //startBin represent bin at which valley finding starts int startBin = hist.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 + //EDGE CASE: Determining if the target direction falls within a bin + //This handled by finding the edges of the valleys if(hist.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 <= hist.getNumBins() && count <= smax) { - //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(hist.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 > smax) //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 smax. + //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 hist.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 + 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 <= hist.getNumBins()/2; i++) { if (hist.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(hist.getValue(i) < valleyThreshold && abs(i - rightIndex) < smax) i++; + int leftIndex = i; + leftTravelDir = (rightIndex + leftIndex)/2; + std::cout << hist.getIndex(leftIndex) << " " << hist.getIndex(rightIndex) << " " << hist.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 < hist.getNumBins()/2) i = hist.getNumBins() - hist.getNumBins()/2; //setting max iteration for left side + i -= hist.getNumBins(); + std::cout << "\ni: " << i << " " << hist.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 << " " << hist.getValue(j) << "\n"; + if (hist.getValue(j) < valleyThreshold) //Found valley + { + int rightIndex = j+1; + //Iterating over valley to find other edge of the valley. + while(hist.getValue(j) < valleyThreshold && abs(j-rightIndex) < smax) + { + j--; + } + int leftIndex = j+1; + leftTravelDir = (rightIndex + leftIndex)/2; + std::cout << hist.getIndex(leftIndex) << " " << hist.getIndex(rightIndex) << " " << hist.getIndex(rightTravelDir) << "\n"; + break; //Since loop begins iterating from the target direction, the valley must be the closest valley + } + } + std::cout << "\nTarget Direction: " << startBin << "\n"; + if(abs(rightTravelDir-startBin) < abs(leftTravelDir-startBin)) { + std::cout << "\nSelected Direction: " << hist.getIndex(rightTravelDir) << "\n"; + return hist.getAngleFromBin(rightTravelDir); + } + else { + std::cout << "\nSelected Direction: " << hist.getIndex(leftTravelDir) << "\n"; + return hist.getAngleFromBin(leftTravelDir); + } } - return hist.getAngleFromBin(travelDirectionIndex); } void printHistogram() diff --git a/main.cpp b/main.cpp index 710d092..9227c3c 100755 --- a/main.cpp +++ b/main.cpp @@ -39,6 +39,8 @@ int main() double speed_init = 0.5; RobotTest bot(init, angle_init, speed_init); + getchar(); + int timestep = 100; diff --git a/map.txt b/map.txt index f43f983..8998813 100644 --- a/map.txt +++ b/map.txt @@ -31,21 +31,21 @@ 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 From fb00ee431ccffb6925eddd72733dc8d58fca511c Mon Sep 17 00:00:00 2001 From: Swapnil Pande Date: Tue, 13 Mar 2018 20:14:14 -0500 Subject: [PATCH 2/5] Latest changes --- main.cpp | 6 ++--- map.txt | 70 ++++++++++++++++++++++++++++---------------------------- map2.txt | 53 ++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 91 insertions(+), 38 deletions(-) create mode 100644 map2.txt diff --git a/main.cpp b/main.cpp index 9227c3c..c451038 100755 --- a/main.cpp +++ b/main.cpp @@ -31,8 +31,8 @@ int main() 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 @@ -52,7 +52,7 @@ int main() bot.talk(); // bot.draw(gnuplot); bot.draw(); - std::this_thread::sleep_for(std::chrono::milliseconds(500)); + std::this_thread::sleep_for(std::chrono::milliseconds(300)); std::cout << "\n\n\n\n\n\n"; } diff --git a/map.txt b/map.txt index 8998813..a717785 100644 --- a/map.txt +++ b/map.txt @@ -1,32 +1,27 @@ 50 50 1 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 @@ -37,15 +32,20 @@ 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 diff --git a/map2.txt b/map2.txt new file mode 100644 index 0000000..6c1964d --- /dev/null +++ b/map2.txt @@ -0,0 +1,53 @@ +50 50 1 +0 0 0 0 0 0 0 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 +0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 + + From 48e999fa16409023fb2f8646681eae5fd1d38604 Mon Sep 17 00:00:00 2001 From: Zhanwen Chen Date: Sat, 17 Mar 2018 16:02:48 -0500 Subject: [PATCH 3/5] features: 1. show robot variables below graph; bug fixes: 1. force correct temporal order of draw() and move() by moving draw() inside move() and privatizing draw(); 2. removed initAngle and initSpeed because it's the robot's job; refactoring: 1. rid some magic numbers; 2. rename instance variables; --- .gitignore | 2 +- Plotter.h | 17 +++-- PolarHistogram.h | 2 +- RobotTest.h | 183 +++++++++++++++++++---------------------------- VFHPather.h | 20 +++--- main.cpp | 44 +++--------- 6 files changed, 104 insertions(+), 164 deletions(-) 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/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 2a37e6a..abc2c12 100755 --- a/PolarHistogram.h +++ b/PolarHistogram.h @@ -108,7 +108,7 @@ class PolarHistogram { { for(int i = 0; i < nBins; i++) { - std::cout << i << " " << getAngleFromBin(i) << " " << histogram[i] << "\n"; + std::cout << i <<"th bin: angle = " << getAngleFromBin(i) << " , certainty = " << histogram[i] << "\n"; } } diff --git a/RobotTest.h b/RobotTest.h index 5ac4389..b812339 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,144 +9,106 @@ #include #include "VFHPather.h" #include "Utils.h" -// #include "gnuplot-iostream.h" #include "Plotter.h" +#define NUM_BINS 16 +#define TARGET_X 50 +#define TARGET_Y 50 +// #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, 200, 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), + hist(NUM_BINS), + pather(hist, &grid, 200, 1, 5, 15), + 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()); - positions.push_back(grid.getTargetLoc()); - 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 = "< i; j--) { - std::cout << "\nj: " << j << " " << i << " " << hist.getValue(j) << "\n"; + // std::cout << "\nj: " << j << " " << i << " " << hist.getValue(j) << "\n"; if (hist.getValue(j) < valleyThreshold) //Found valley { int rightIndex = j+1; @@ -193,17 +193,17 @@ class VFHPather{ } int leftIndex = j+1; leftTravelDir = (rightIndex + leftIndex)/2; - std::cout << hist.getIndex(leftIndex) << " " << hist.getIndex(rightIndex) << " " << hist.getIndex(rightTravelDir) << "\n"; + // 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 = 0; init.y = 0; // double angle_init = 0.78; // pi/4 - double angle_init = 90; // pi/4 - double speed_init = 0.5; + // double angle_init = 90; // pi/4 + // double speed_init = 0.5; - RobotTest bot(init, angle_init, speed_init); + RobotTest bot(init); getchar(); + int numTimesteps = 100; - int timestep = 100; - - - for(int i = 0; i < timestep; ++i) + for(int i = 0; i < numTimesteps; ++i) { - std::cout< Date: Tue, 20 Mar 2018 20:48:17 -0500 Subject: [PATCH 4/5] tried to refactor but broke the polar histogram --- HistogramGrid.h | 43 ++++++++++---------- PolarHistogram.h | 25 +++++------- README.md | 19 ++++++++- RobotTest.h | 23 +++++------ VFHPather.h | 102 ++++++++++++++++++++++++++++++----------------- main.cpp | 36 +++++++++++++---- map.txt | 3 -- map2.txt | 3 -- 8 files changed, 156 insertions(+), 98 deletions(-) diff --git a/HistogramGrid.h b/HistogramGrid.h index 3557b3f..097897b 100755 --- a/HistogramGrid.h +++ b/HistogramGrid.h @@ -1,7 +1,6 @@ // // Created by Swapnil on 1/25/2018. // - #ifndef VECTORFIELDHISTOGRAMTESTING_HISTOGRAMGRID_H #define VECTORFIELDHISTOGRAMTESTING_HISTOGRAMGRID_H @@ -12,6 +11,7 @@ #include "Utils.h" + class HistogramGrid { private: int iMax; //Size in the i direction of the histogram grid @@ -33,11 +33,18 @@ class HistogramGrid { //Creates a new histogram grid object with no objects present in the grid // 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) + // double nodeSideLen - Side dimension of each node. histWidth and histLength should be divisible by this number + // int activeRegionSize_i + // int activeRegionSize_j + 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); // 50 + // + // 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/PolarHistogram.h b/PolarHistogram.h index abc2c12..2ec0d36 100755 --- a/PolarHistogram.h +++ b/PolarHistogram.h @@ -15,7 +15,10 @@ class PolarHistogram { public: //Creates a Polar Histogram object with the number of bins passed - PolarHistogram(int numBins): nBins(numBins), binWidth(360.0/numBins), histogram(new double[nBins]) + PolarHistogram(int numBins): + nBins(numBins), + binWidth(360.0/numBins), + histogram(new double[nBins]) { for(int i = 0; i < nBins; i++) { @@ -23,22 +26,15 @@ class PolarHistogram { } } - //PolarHistogram - //Testing constructor with hardcoded values - PolarHistogram(bool test): nBins(10), binWidth(36.0), histogram(new double[10]) + void setPolarHistogramValue(int bin, double newValue) { - for(int i = 0; i < 3; i ++) + if (bin < nBins) { - histogram[i] = 0; + histogram[bin] = newValue; + } else + { + throw "PolarHistogram.setPolarHistogramValue: given bin out of range"; } - - histogram[3] = 7; - histogram[4] = 7; - histogram[5] = 3; - histogram[6] = 3; - histogram[7] = 10; - histogram[8] = 2; - histogram[9] = 3; } //getIndex @@ -73,7 +69,6 @@ class PolarHistogram { //Returns the value of the histogram for the specified bin double getValue(int bin) { - return histogram[getIndex(bin)]; } diff --git a/README.md b/README.md index 82057f1..f9697b1 100755 --- a/README.md +++ b/README.md @@ -1 +1,18 @@ -"# VectorFieldHistogramTesting" +# VectorFieldHistogramTesting + +This is a motion planning software with the Vector Field Histogram algorithm proposed by Johann Borenstein and Yoram Koren in 1991. The software comes with a 2D simulation with GNUPlot (Plotter.h). + +Entry Point: main.cpp + +### Run Instructions: + +``` +cd build +cmake .. +make +./VectorFieldHistogramTesting +``` + +Then press any key to start the simulation + +### diff --git a/RobotTest.h b/RobotTest.h index b812339..c4a8907 100644 --- a/RobotTest.h +++ b/RobotTest.h @@ -11,11 +11,6 @@ #include "Utils.h" #include "Plotter.h" -#define NUM_BINS 16 -#define TARGET_X 50 -#define TARGET_Y 50 -// #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. @@ -24,11 +19,10 @@ class RobotTest{ private: HistogramGrid grid; - PolarHistogram hist; + PolarHistogram polarHist; VFHPather pather; //Object used to store the grid/map of obstacles 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; @@ -85,14 +79,19 @@ class 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), - hist(NUM_BINS), - pather(hist, &grid, 200, 1, 5, 15), + RobotTest(discretePoint initPos, discretePoint targetPos, int numBins, + int iSizeActiveRegion, int jSizeActiveRegion, int histWidth, + int histLength, int nodeSize, int maxNumNodesForValley, + double a, double b, int l, double valleyThreshold): + grid("../map.txt", initPos, iSizeActiveRegion, jSizeActiveRegion, histWidth, + histLength, nodeSize), + polarHist(numBins), + pather(polarHist, &grid, iSizeActiveRegion, jSizeActiveRegion, histWidth, + histLength, nodeSize, maxNumNodesForValley, a, b, l, valleyThreshold), location(initPos) { pather.updateRobotPosition(location); - grid.setTargetLoc({TARGET_X, TARGET_Y}); + grid.setTargetLoc({targetPos.x, targetPos.y}); } diff --git a/VFHPather.h b/VFHPather.h index cd5cb06..f72518d 100755 --- a/VFHPather.h +++ b/VFHPather.h @@ -1,6 +1,7 @@ // // Created by Swapnil on 2/3/2018. -// +// REVIEW: bug source: if multiple valid directions, which one to choose? +// REVIEW: Not considering far enough obstacles (activeRegion)? #ifndef VECTORFIELDHISTOGRAMTESTING_VFHPATHER_H #define VECTORFIELDHISTOGRAMTESTING_VFHPATHER_H @@ -16,24 +17,34 @@ class VFHPather{ private: - - PolarHistogram hist; //Object used to store the polar histogram + PolarHistogram polarHist; //Object used to store the polar histogram HistogramGrid* grid; //Object used to store the grid/map of obstacles - - double a; //Constants for the weighting function for histogram generation + //Constants for the weighting function for histogram generation + double a; double b; int l; //Smoothing constant for polar histogram - int smax; //Maximum number of nodes that define a wide valley + int maxNumNodesForValley; //Maximum number of nodes that define a wide valley double valleyThreshold; + double getRobotToTargetAngle() + { + //startBin represent bin at which valley finding starts + discretePoint robotLocation = (*grid).getRobotLoc(); + discretePoint targetLocation = grid->getTargetLoc(); + + // REVIEW: is the angle relative or absolute? + double robotToTargetAngle = (*grid).getAngle(robotLocation, targetLocation); + return robotToTargetAngle; + } + public: //Default Constructor, implements testing mode of Pather with hard-coded default values - VFHPather(): hist(true), grid(new HistogramGrid(10, 10, 0.1)), a(50), b(100), l(5), valleyThreshold(15) - {} + // VFHPather(): hist(true), grid(new HistogramGrid(10, 10, 0.1)), a(50), b(100), l(5), valleyThreshold(15) + // {} //Alternate constructor takes all parameters //histIn - PolarHistogram object to store the histogram @@ -43,9 +54,21 @@ class VFHPather{ //lIn - Smoothing algorithm constant - determines the number of bins included in smoothing //valleyThreshold - Threshold for determining whether bin should be included in valley // If polar object density falls below valley threshold, bin is considered to be part of valley - VFHPather(PolarHistogram &histIn, HistogramGrid *gridIn, double aIn, double bIn, - double lIn, double valleyThresholdIn): hist(histIn), grid(gridIn), a(aIn), b(bIn), l(lIn), - smax(10), valleyThreshold(valleyThresholdIn) {} + VFHPather(PolarHistogram &histIn, HistogramGrid *gridIn, + int iSizeActiveRegion, int jSizeActiveRegion, int histWidth, + int histLength, double nodeSize, + int maxNumNodesForValley, double aIn, double bIn, + double lIn, double valleyThresholdIn): + polarHist(histIn), + grid(gridIn), + a(aIn), + b(bIn), + l(lIn), + maxNumNodesForValley(maxNumNodesForValley), + valleyThreshold(valleyThresholdIn) + { + + } //TODO: Add ability to dynamically set certainty value //TODO This function may be deprecated as we restructure the robot code for ROSMOD //updateRobotPosition @@ -54,16 +77,13 @@ class VFHPather{ (*grid).setRobotLoc(pos); } - //generateHistogram //Builds the vector field histogram based on current position of robot and surrounding obstacles void generateHistogram() { + polarHist.reset(); //Resetting the histogram - - hist.reset(); //Resetting the histogram - - region activeRegion = (*grid).getActiveRegion(); + region activeRegion = (*grid).getActiveRegion(); discretePoint curNode; //Node currently being iterated over @@ -72,7 +92,7 @@ class VFHPather{ for(curNode.y = activeRegion.min.y; curNode.y < activeRegion.max.y; curNode.y++) { double val = pow( (*grid).getCertainty(curNode),2)*(a-b* (*grid).getDistance(curNode, (*grid).getRobotLoc())); - hist.addValue( (*grid).getAngle( (*grid).getRobotLoc(), curNode), val); + polarHist.addValue( (*grid).getAngle( (*grid).getRobotLoc(), curNode), val); //std::cout << curNode.x << " " << curNode.y << " " << val << "\n"; (*grid).getCertainty(curNode); @@ -96,14 +116,14 @@ class VFHPather{ generateHistogram(); //Computes the polar histogram printHistogram(); - //hist.smoothHistogram(l); //Smoothing histogram + //hist.smoothHistogram(l); //Smoothing histogram REVIEW: should uncomment this? + double robotToTargetAngle = getRobotToTargetAngle(); - //startBin represent bin at which valley finding starts - int startBin = hist.getBinFromAngle((*grid).getAngle((*grid).getRobotLoc(), grid->getTargetLoc())); //Determine the bin in which the target falls + int startBin = polarHist.getBinFromAngle(robotToTargetAngle); //Determine the bin in which the target falls //EDGE CASE: Determining if the target direction falls within a bin //This handled by finding the edges of the valleys - if(hist.getValue(startBin) < valleyThreshold) //Desired travel direction falls within a valley + if(polarHist.getValue(startBin) < valleyThreshold) //Desired travel direction falls within a valley { std::cout << "\n\n\nREACHED EDGE CASE\n\n\n"; int leftIndex = -1, rightIndex = -1; //Store the indices of the edges of the valley @@ -115,12 +135,11 @@ class VFHPather{ int i; //Stores index to search //Iterating over the histogram to find valley. //Iteration occurs alternating in left & right direction of target - while(count <= hist.getNumBins() && count <= smax) + while(count <= polarHist.getNumBins() && count <= maxNumNodesForValley) { - i = startBin + negative * count / divide; //Index of bin to check next - if(hist.getValue(i) > valleyThreshold) + if(polarHist.getValue(i) > valleyThreshold) { if(negative == 1) rightIndex = i + 1; else leftIndex = i; @@ -135,9 +154,9 @@ class VFHPather{ count++; //Increment counter } - if(count > smax) //The maximum size of a valley was reached. Write the edges at the last searched bins + if(count > maxNumNodesForValley) //The maximum size of a valley was reached. Write the edges at the last searched bins { - //Stores the edges of the valley if the size has reached smax. + //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++) @@ -151,7 +170,7 @@ class VFHPather{ } // std::cout << leftIndex << " " << rightIndex << "\n"; //Returns the average value of the bins - return hist.getAngleFromBin((rightIndex+leftIndex)/2); + return polarHist.getAngleFromBin((rightIndex+leftIndex)/2); } else //Not edge case, process histogram normally { @@ -162,12 +181,12 @@ class VFHPather{ int i; //Checking left side // std::cout << "\n Left Side parameters: \n"; - for (i = startBin + 1; i <= hist.getNumBins()/2; i++) { - if (hist.getValue(i) < valleyThreshold) //Found valley + for (i = startBin + 1; i <= polarHist.getNumBins()/2; i++) { + if (polarHist.getValue(i) < valleyThreshold) //Found valley { int rightIndex = i; //Iterating over valley to find other edge of the valley. - while(hist.getValue(i) < valleyThreshold && abs(i - rightIndex) < smax) i++; + while(polarHist.getValue(i) < valleyThreshold && abs(i - rightIndex) < maxNumNodesForValley) i++; // REVIEW int leftIndex = i; leftTravelDir = (rightIndex + leftIndex)/2; // std::cout <<"lol"<< hist.getIndex(leftIndex) << " " << hist.getIndex(rightIndex) << " " << hist.getIndex(leftTravelDir) << "\n"; @@ -175,19 +194,28 @@ class VFHPather{ } } - if(i < hist.getNumBins()/2) i = hist.getNumBins() - hist.getNumBins()/2; //setting max iteration for left side - i -= hist.getNumBins(); + // if(i < hist.getNumBins()/2) i = hist.getNumBins() - hist.getNumBins()/2; //setting max iteration for left side + + //setting max iteration for left side + int numBins = polarHist.getNumBins(); + if(i < numBins/2) + { + i = numBins/2; + } + + // i -= hist.getNumBins(); + i -= numBins; // std::cout << "\ni: " << i << " " << hist.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 << " " << hist.getValue(j) << "\n"; - if (hist.getValue(j) < valleyThreshold) //Found valley + if (polarHist.getValue(j) < valleyThreshold) //Found valley { int rightIndex = j+1; //Iterating over valley to find other edge of the valley. - while(hist.getValue(j) < valleyThreshold && abs(j-rightIndex) < smax) + while(polarHist.getValue(j) < valleyThreshold && abs(j-rightIndex) < maxNumNodesForValley) { j--; } @@ -200,18 +228,18 @@ class VFHPather{ // std::cout << "\nTarget Direction: " << startBin << "\n"; if(abs(rightTravelDir-startBin) < abs(leftTravelDir-startBin)) { // std::cout << "\nSelected Direction: " << hist.getIndex(rightTravelDir) << "\n"; - return hist.getAngleFromBin(rightTravelDir); + return polarHist.getAngleFromBin(rightTravelDir); } else { // std::cout << "\nSelected Direction: " << hist.getIndex(leftTravelDir) << "\n"; - return hist.getAngleFromBin(leftTravelDir); + return polarHist.getAngleFromBin(leftTravelDir); } } } void printHistogram() { - hist.printHistogram(); + polarHist.printHistogram(); } int** getObjectGrid() diff --git a/main.cpp b/main.cpp index 99f2308..f9068d8 100755 --- a/main.cpp +++ b/main.cpp @@ -10,21 +10,43 @@ #include "Plotter.h" +// BUG: 1. speed updating is not correct: why is it moving by 1 block every time? +// BUG: 2. simulation stopped at 98. Why? +// BUG: 3. robot going through obstacles. Why? + int getIndex(int a, int n) { return ((a % n) + n) % n; } int main() { - discretePoint init; - init.x = 0; - init.y = 0; + // Valley Hyperparameters + int maxNumNodesForValley = 10; + double a = 200; + double b = 1; + int l = 5; + double valleyThreshold = 15; + + // Polar Hyperparameters + int numBins = 16; + int iSizeActiveRegion = 20; + int jSizeActiveRegion = 20; + int histWidth = 50; + int histLength = 50; + double nodeSize = 1; + + + discretePoint initPosition; + initPosition.x = 0; + initPosition.y = 0; + + discretePoint targetPosition; + targetPosition.x = 50; + targetPosition.y = 50; - // double angle_init = 0.78; // pi/4 - // double angle_init = 90; // pi/4 - // double speed_init = 0.5; + RobotTest bot(initPosition, targetPosition, numBins, iSizeActiveRegion, jSizeActiveRegion, histWidth, + histLength, nodeSize, maxNumNodesForValley, a, b, l, valleyThreshold); - RobotTest bot(init); getchar(); int numTimesteps = 100; diff --git a/map.txt b/map.txt index a717785..6cd7391 100644 --- a/map.txt +++ b/map.txt @@ -1,4 +1,3 @@ -50 50 1 0 0 0 0 0 0 0 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 @@ -49,5 +48,3 @@ 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 - - diff --git a/map2.txt b/map2.txt index 6c1964d..755cf2e 100644 --- a/map2.txt +++ b/map2.txt @@ -1,4 +1,3 @@ -50 50 1 0 0 0 0 0 0 0 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 @@ -49,5 +48,3 @@ 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 - - From 206705683b5ff9df525f11f4af08685d2e327034 Mon Sep 17 00:00:00 2001 From: Zhanwen Chen Date: Thu, 29 Mar 2018 20:01:22 -0500 Subject: [PATCH 5/5] (refactor) parameterized all magic numbers into the top of RobotTest --- HistogramGrid.h | 23 +++++++++---- PolarHistogram.h | 25 ++++++++------ README.md | 18 ++-------- RobotTest.h | 36 +++++++++++++------- VFHPather.h | 86 ++++++++++++++++++++++-------------------------- main.cpp | 40 +++++----------------- map2.txt | 3 ++ map_backup.txt | 51 ++++++++++++++++++++++++++++ run | 3 ++ 9 files changed, 165 insertions(+), 120 deletions(-) create mode 100644 map_backup.txt create mode 100755 run diff --git a/HistogramGrid.h b/HistogramGrid.h index 097897b..c0dfa67 100755 --- a/HistogramGrid.h +++ b/HistogramGrid.h @@ -1,6 +1,7 @@ // // Created by Swapnil on 1/25/2018. // + #ifndef VECTORFIELDHISTOGRAMTESTING_HISTOGRAMGRID_H #define VECTORFIELDHISTOGRAMTESTING_HISTOGRAMGRID_H @@ -11,7 +12,6 @@ #include "Utils.h" - class HistogramGrid { private: int iMax; //Size in the i direction of the histogram grid @@ -33,9 +33,7 @@ class HistogramGrid { //Creates a new histogram grid object with no objects present in the grid // int histWidth - Width of the entire histogram in meters // int histLength - Length of the entire histogram in meters - // double nodeSideLen - Side dimension of each node. histWidth and histLength should be divisible by this number - // int activeRegionSize_i - // int activeRegionSize_j + // int nodeSideLen - Side dimension of each node. histWidth and histLength should be divisible by this number HistogramGrid(int histWidth, int histLength, double nodeSideLen, int activeRegionSize_i, int activeRegionSize_j): iMax((int)(histWidth/nodeSideLen)), @@ -67,19 +65,30 @@ class HistogramGrid { } //HistogramGrid - // BUG: activeRegionSize_i, activeRegionSize_i are not used here //Alternate constructor used for ingesting grid from file. Used only for testing - HistogramGrid(std::string fName, discretePoint robotLocIn, int iSizeActiveRegion, int jSizeActiveRegion, int histWidth, int histLength, double nodeSize) + // HistogramGrid(std::string fName, discretePoint robotLocIn, + // int activeRegionSize_i, int activeRegionSize_j): + // iSizeActiveRegion(activeRegionSize_i), + // jSizeActiveRegion(activeRegionSize_j) + HistogramGrid(std::string fName, discretePoint robotLocIn, + int activeRegionSize_i, int activeRegionSize_j, + int histWidth, int histLength, double nodeSizeIn): + iSizeActiveRegion(activeRegionSize_i), + jSizeActiveRegion(activeRegionSize_j), + nodeSize(nodeSizeIn) { //std::cout << "\n\ntesting histogram grid initialization: "<< std::endl; + robotLoc = robotLocIn; std::string data; //Temporary string to store ingested data + // int histWidth; + // int histLength; std::ifstream file(fName); if (file.is_open()) { // file >> data; - // histWidth = std::stoi(data); // 50 + // histWidth = std::stoi(data); // // file >> data; // histLength = std::stoi(data); diff --git a/PolarHistogram.h b/PolarHistogram.h index 2ec0d36..abc2c12 100755 --- a/PolarHistogram.h +++ b/PolarHistogram.h @@ -15,10 +15,7 @@ class PolarHistogram { public: //Creates a Polar Histogram object with the number of bins passed - PolarHistogram(int numBins): - nBins(numBins), - binWidth(360.0/numBins), - histogram(new double[nBins]) + PolarHistogram(int numBins): nBins(numBins), binWidth(360.0/numBins), histogram(new double[nBins]) { for(int i = 0; i < nBins; i++) { @@ -26,15 +23,22 @@ class PolarHistogram { } } - void setPolarHistogramValue(int bin, double newValue) + //PolarHistogram + //Testing constructor with hardcoded values + PolarHistogram(bool test): nBins(10), binWidth(36.0), histogram(new double[10]) { - if (bin < nBins) + for(int i = 0; i < 3; i ++) { - histogram[bin] = newValue; - } else - { - throw "PolarHistogram.setPolarHistogramValue: given bin out of range"; + histogram[i] = 0; } + + histogram[3] = 7; + histogram[4] = 7; + histogram[5] = 3; + histogram[6] = 3; + histogram[7] = 10; + histogram[8] = 2; + histogram[9] = 3; } //getIndex @@ -69,6 +73,7 @@ class PolarHistogram { //Returns the value of the histogram for the specified bin double getValue(int bin) { + return histogram[getIndex(bin)]; } diff --git a/README.md b/README.md index f9697b1..fad6faa 100755 --- a/README.md +++ b/README.md @@ -1,18 +1,6 @@ # VectorFieldHistogramTesting -This is a motion planning software with the Vector Field Histogram algorithm proposed by Johann Borenstein and Yoram Koren in 1991. The software comes with a 2D simulation with GNUPlot (Plotter.h). +Bugs -Entry Point: main.cpp - -### Run Instructions: - -``` -cd build -cmake .. -make -./VectorFieldHistogramTesting -``` - -Then press any key to start the simulation - -### +- [] contpoint vs discretepoint angle issue. Error may be due to discretization. +- [] smoothing? diff --git a/RobotTest.h b/RobotTest.h index c4a8907..d704b6e 100644 --- a/RobotTest.h +++ b/RobotTest.h @@ -11,6 +11,24 @@ #include "Utils.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. @@ -19,10 +37,11 @@ class RobotTest{ private: HistogramGrid grid; - PolarHistogram polarHist; + PolarHistogram hist; VFHPather pather; //Object used to store the grid/map of obstacles 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; @@ -79,19 +98,14 @@ class 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, discretePoint targetPos, int numBins, - int iSizeActiveRegion, int jSizeActiveRegion, int histWidth, - int histLength, int nodeSize, int maxNumNodesForValley, - double a, double b, int l, double valleyThreshold): - grid("../map.txt", initPos, iSizeActiveRegion, jSizeActiveRegion, histWidth, - histLength, nodeSize), - polarHist(numBins), - pather(polarHist, &grid, iSizeActiveRegion, jSizeActiveRegion, histWidth, - histLength, nodeSize, maxNumNodesForValley, a, b, l, valleyThreshold), + 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) { pather.updateRobotPosition(location); - grid.setTargetLoc({targetPos.x, targetPos.y}); + grid.setTargetLoc({TARGET_X, TARGET_Y}); } diff --git a/VFHPather.h b/VFHPather.h index f72518d..cb4118e 100755 --- a/VFHPather.h +++ b/VFHPather.h @@ -1,7 +1,6 @@ // // Created by Swapnil on 2/3/2018. -// REVIEW: bug source: if multiple valid directions, which one to choose? -// REVIEW: Not considering far enough obstacles (activeRegion)? +// #ifndef VECTORFIELDHISTOGRAMTESTING_VFHPATHER_H #define VECTORFIELDHISTOGRAMTESTING_VFHPATHER_H @@ -17,11 +16,11 @@ class VFHPather{ private: + PolarHistogram polarHist; //Object used to store the polar histogram HistogramGrid* grid; //Object used to store the grid/map of obstacles - //Constants for the weighting function for histogram generation - double a; + double a; //Constants for the weighting function for histogram generation double b; int l; //Smoothing constant for polar histogram @@ -30,20 +29,25 @@ class VFHPather{ double valleyThreshold; - double getRobotToTargetAngle() - { - //startBin represent bin at which valley finding starts - discretePoint robotLocation = (*grid).getRobotLoc(); - discretePoint targetLocation = grid->getTargetLoc(); - - // REVIEW: is the angle relative or absolute? - double robotToTargetAngle = (*grid).getAngle(robotLocation, targetLocation); - return robotToTargetAngle; - } - public: + // Preferred constructor + // VFHPather(PolarHistogram &histIn, HistogramGrid *gridIn, + // int iSizeActiveRegion, int jSizeActiveRegion, int histWidth, + // int histLength, double nodeSize, + // int maxNumNodesForValley, double aIn, double bIn, + // double lIn, double valleyThresholdIn): + // polarHist(histIn), + // grid(gridIn), + // a(aIn), + // b(bIn), + // l(lIn), + // maxNumNodesForValley(maxNumNodesForValley), + // valleyThreshold(valleyThresholdIn) + // { + // std::cout<<"VFHPather: Using preferred constructor."; + // } //Default Constructor, implements testing mode of Pather with hard-coded default values - // VFHPather(): hist(true), grid(new HistogramGrid(10, 10, 0.1)), a(50), b(100), l(5), valleyThreshold(15) + // VFHPather(): polarHist(true), grid(new HistogramGrid(10, 10, 0.1)), a(50), b(100), l(5), valleyThreshold(15) // {} //Alternate constructor takes all parameters @@ -54,20 +58,17 @@ class VFHPather{ //lIn - Smoothing algorithm constant - determines the number of bins included in smoothing //valleyThreshold - Threshold for determining whether bin should be included in valley // If polar object density falls below valley threshold, bin is considered to be part of valley - VFHPather(PolarHistogram &histIn, HistogramGrid *gridIn, - int iSizeActiveRegion, int jSizeActiveRegion, int histWidth, - int histLength, double nodeSize, - int maxNumNodesForValley, double aIn, double bIn, - double lIn, double valleyThresholdIn): + VFHPather(PolarHistogram &histIn, HistogramGrid *gridIn, double aIn, double bIn, + double lIn, double valleyThresholdIn, int maxNumNodesForValleyIn): polarHist(histIn), grid(gridIn), a(aIn), b(bIn), l(lIn), - maxNumNodesForValley(maxNumNodesForValley), + maxNumNodesForValley(maxNumNodesForValleyIn), valleyThreshold(valleyThresholdIn) { - + std::cout<<"VFHPather: Using secondary constructor.\n"; } //TODO: Add ability to dynamically set certainty value //TODO This function may be deprecated as we restructure the robot code for ROSMOD @@ -77,13 +78,14 @@ class VFHPather{ (*grid).setRobotLoc(pos); } + //generateHistogram //Builds the vector field histogram based on current position of robot and surrounding obstacles void generateHistogram() { polarHist.reset(); //Resetting the histogram - region activeRegion = (*grid).getActiveRegion(); + region activeRegion = (*grid).getActiveRegion(); discretePoint curNode; //Node currently being iterated over @@ -116,10 +118,10 @@ class VFHPather{ generateHistogram(); //Computes the polar histogram printHistogram(); - //hist.smoothHistogram(l); //Smoothing histogram REVIEW: should uncomment this? - double robotToTargetAngle = getRobotToTargetAngle(); + //polarHist.smoothHistogram(l); //Smoothing histogram - int startBin = polarHist.getBinFromAngle(robotToTargetAngle); //Determine the bin in which the target falls + //startBin represent bin at which valley finding starts + int startBin = polarHist.getBinFromAngle((*grid).getAngle((*grid).getRobotLoc(), grid->getTargetLoc())); //Determine the bin in which the target falls //EDGE CASE: Determining if the target direction falls within a bin //This handled by finding the edges of the valleys @@ -137,6 +139,7 @@ class VFHPather{ //Iteration occurs alternating in left & right direction of target while(count <= polarHist.getNumBins() && count <= maxNumNodesForValley) { + i = startBin + negative * count / divide; //Index of bin to check next if(polarHist.getValue(i) > valleyThreshold) @@ -186,52 +189,43 @@ class VFHPather{ { int rightIndex = i; //Iterating over valley to find other edge of the valley. - while(polarHist.getValue(i) < valleyThreshold && abs(i - rightIndex) < maxNumNodesForValley) i++; // REVIEW + while(polarHist.getValue(i) < valleyThreshold && abs(i - rightIndex) < maxNumNodesForValley) i++; int leftIndex = i; leftTravelDir = (rightIndex + leftIndex)/2; - // std::cout <<"lol"<< hist.getIndex(leftIndex) << " " << hist.getIndex(rightIndex) << " " << hist.getIndex(leftTravelDir) << "\n"; + // 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 } } - // if(i < hist.getNumBins()/2) i = hist.getNumBins() - hist.getNumBins()/2; //setting max iteration for left side - - //setting max iteration for left side - int numBins = polarHist.getNumBins(); - if(i < numBins/2) - { - i = numBins/2; - } - - // i -= hist.getNumBins(); - i -= numBins; - // std::cout << "\ni: " << i << " " << hist.getIndex(i) << std::endl; + 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 << " " << hist.getValue(j) << "\n"; + // 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) + while(polarHist.getValue(j) < valleyThreshold && abs(j-rightIndex) < maxNumNodesForValley) { j--; } int leftIndex = j+1; leftTravelDir = (rightIndex + leftIndex)/2; - // std::cout << "lol"<