Skip to content

Commit e82315e

Browse files
committed
more quadtrees test + some minor bugs correction
1 parent d68a7f8 commit e82315e

8 files changed

+151
-87
lines changed

CMakeLists.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ set(BOOST_REQUIRED_VERSION 1.62.0)
66
set(EXE_NAME "packer")
77

88
# Compiler flags
9-
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Wextra -Wfatal-errors -std=c++11 -fopenmp")
9+
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Wextra -Wfatal-errors -std=c++11 -O0 -g -fopenmp")
1010

1111
# Options
1212
option(FORCE_UNPACK "Boost unpack forcing" 0) #Set to 1 to force local installation of Boost (ensure that it will take priority)

src/Shape.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -245,7 +245,6 @@ Point Shape::centroid() const {
245245
* @param s Another shape
246246
*/
247247
bool Shape::intersectsWith(const Shape& s) const {
248-
cerr << "Shape called" << endl;
249248
return bg::overlaps(_multiP, s._multiP);
250249
}
251250

src/Shape.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ class Shape {
7373
virtual void translate(double Tx, double Ty);
7474
virtual void envelope(Box&) const;
7575
virtual Point centroid() const;
76-
virtual int area() const;
76+
virtual int area() const;
7777
virtual bool intersectsWith(const Shape&) const;
7878
virtual bool intersectsWith(const Ring& s) const;
7979
virtual void convexHull(Polygon&) const;

src/quadtree/InnerQuadTree.cpp

+5-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
#include <iostream>
22
#include <cmath>
3+
#include <iomanip>
34

45
#include <boost/geometry.hpp>
56
// Plante sans cette include pour une raison indeterminé
@@ -151,8 +152,7 @@ bool InnerQuadTree::intersectsRec(const InnerQuadTree& q, float offsetX1, float
151152
return false;
152153
*/
153154
Box b1{{x1 + offsetX1, y1 + offsetY1}, {x2 + offsetX1, y2 + offsetY1}};
154-
Box b2{{q.x1 + offsetX2, q.y1 + offsetY2}, {q.x2 + offsetX2, q.y2 + offsetY2}};
155-
cout << bg::wkt(b1) << ":" << bg::wkt(b2) << endl;
155+
Box b2{{q.x1 + offsetX2, q.y1 + offsetY2}, {q.x2 + offsetX2, q.y2 + offsetY2}};
156156
bool boxIntersects = bg::intersects(b1, b2);
157157
//cout << boxIntersects << test << endl;
158158

@@ -265,8 +265,9 @@ void InnerQuadTree::translater(float x, float y) {
265265

266266

267267
std::ostream& operator<<(std::ostream& s, const InnerQuadTree& q) {
268-
s << "Boundind box : (" << q.x1 << "," << q.y1 << ")-(" << q.x2 << "," << q.y2 << ") " <<
269-
"Size : " << q.x2 - q.x1 << "," << q.y2 - q.y1 << " InnerTrees : " << q.size;
268+
s << fixed << setprecision(2);
269+
s << setw(6) << "Boundind box : (" << setw(6) << q.x1 << "," << setw(6) << q.y1 << ")-(" << setw(6) << q.x2 << "," << setw(6) << q.y2 << ") " <<
270+
"Size : (" << setw(8) << (q.x2 - q.x1) << "," << setw(8) << (q.y2 - q.y1) << ") InnerTrees : " << q.size;
270271
return s;
271272
}
272273

src/quadtree/QuadTree.cpp

+38-35
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@
44
#include <fstream>
55

66

7+
#include <boost/geometry/algorithms/area.hpp>
8+
#include <boost/geometry/strategies/cartesian/area_surveyor.hpp>
79
#include <boost/geometry/algorithms/correct.hpp>
810
#include <boost/geometry/algorithms/intersects.hpp>
911
#include <boost/math/constants/constants.hpp>
@@ -33,7 +35,8 @@ void QuadTree::copy(const QuadTree& q) {
3335
_totalX = q._totalX;
3436
_totalY = q._totalY;
3537
_maxDepth = q._maxDepth;
36-
precision = q.precision;
38+
precision = q.precision;
39+
_area = q._area;
3740
}
3841

3942
/**
@@ -70,7 +73,8 @@ QuadTree& QuadTree::operator=(QuadTree&& q) {
7073
_totalX = q._totalX;
7174
_totalY = q._totalY;
7275
_maxDepth = q._maxDepth;
73-
precision = q.precision;
76+
precision = q.precision;
77+
_area = q._area;
7478
}
7579

7680
return *this;
@@ -113,7 +117,8 @@ QuadTree::QuadTree(Shape& s, float precision) :
113117
than precision can be detected by the quadtree
114118
*/
115119
QuadTree::QuadTree(MultiPolygon& mult, float precision, unsigned id)
116-
: Shape(mult, id), trees(nullptr), precision(precision) {
120+
: Shape(mult, id), trees(nullptr), precision(precision) {
121+
_area = bg::area(mult);
117122
float rotationAngle = 360.f / quadsNumber;
118123
float currentAngle = 0.f;
119124
trees = new InnerQuadTree*[quadsNumber];
@@ -175,55 +180,52 @@ QuadTree::QuadTree(MultiPolygon& mult, float precision, unsigned id)
175180
* @param q
176181
* @return
177182
*/
178-
bool QuadTree::intersectsWith(const QuadTree& q) const {
179-
cerr << "QuadTree instersect called" << endl;
183+
bool QuadTree::intersectsWith(const Shape& s) const {
184+
const QuadTree& q = static_cast<const QuadTree &>(s);
180185
return trees[currentTree]->intersectsRec(*q.trees[q.currentTree], _totalX, _totalY,
181186
q._totalX, q._totalY);
182187
}
183188

184189
/**
185-
* @brief QuadTree::translater translates the QuadTree position
186-
* @param x
187-
* @param y
188-
*/
189-
void QuadTree::translater(float x, float y) {
190-
_totalX += x;
191-
_totalY += y;
192-
}
193-
194-
/**
195-
* @brief QuadTree::rotater rotate a QuadTree
190+
* @brief QuadTree::rotate overload of Shape::rotate for quadtrees
191+
* rotate a QuadTree
196192
* The rotation is around the (0,0) point by default
197193
* Better rotation is possible with the possibility to choose the rotation point
198194
* @param angle rotation angle in degree, stored in radians
199195
*/
200-
void QuadTree::rotater(float angle) {
201-
unsigned newQuad = (int) round(angle * quadsNumber / 360) % quadsNumber;
202-
float newAngle = pi * angle / 180.f;
203-
// Compute the tree position
204-
float newX = cos(newAngle) * _totalX - sin(newAngle) * _totalY;
205-
float newY = sin(newAngle) * _totalX + cos(newAngle) * _totalY;
206-
newX += treesOffset[newQuad].x();
207-
newY += treesOffset[newQuad].y();
208-
currentTree = newQuad;
209-
}
210-
211-
/**
212-
* @brief QuadTree::rotate overload of Shape::rotate for quadtrees
213-
* @param x
214-
* @param y
215-
*/
216-
void QuadTree::rotate(double angle) {
217-
rotater(angle);
196+
void QuadTree::rotate(double degrees) {
197+
float anglePrecision = 360.0 / quadsNumber;
198+
float angle = static_cast<float>(degrees) + anglePrecision*currentTree;
199+
_totalX -= treesOffset[currentTree].x();
200+
_totalY -= treesOffset[currentTree].y();
201+
202+
unsigned newQuad = (int) round(angle / anglePrecision) % quadsNumber;
203+
float newAngle = pi * angle / 180.f;
204+
// Compute the tree position
205+
float newX = cos(newAngle) * _totalX - sin(newAngle) * _totalY;
206+
float newY = sin(newAngle) * _totalX + cos(newAngle) * _totalY;
207+
_totalX = newX + treesOffset[newQuad].x();
208+
_totalY = newY + treesOffset[newQuad].y();
209+
currentTree = newQuad;
218210
}
219211

220212
/**
221213
* @brief QuadTree::translate overload of Shape::translate for quadtrees
214+
* translates the QuadTree position
222215
* @param x
223216
* @param y
224217
*/
225218
void QuadTree::translate(double Tx, double Ty) {
226-
translater(Tx, Ty);
219+
_totalX += static_cast<float>(Tx);
220+
_totalY += static_cast<float>(Ty);
221+
}
222+
223+
void QuadTree::envelope(Box& b) const {
224+
b = {{_totalX,_totalY},{_totalX+trees[currentTree]->x2,_totalY+trees[currentTree]->y2}};
225+
}
226+
227+
int QuadTree::area() const {
228+
return _area;
227229
}
228230

229231
/**
@@ -234,6 +236,7 @@ void QuadTree::translate(double Tx, double Ty) {
234236
std::ostream& operator <<(std::ostream& s, const QuadTree& q) {
235237
s << "QuadTree n° : " << q.getIdentifier() << endl;
236238
s << "Absolute Position : (" << q._totalX << "," << q._totalY << ") " << endl;
239+
s << "Current tree (angle) : " << q.currentTree << "(" << 30*q.currentTree << ")" << endl;
237240

238241
s << "Trees : " << std::endl;
239242
for (unsigned i = 0; i < q.quadsNumber; i++)

src/quadtree/QuadTree.hpp

+8-9
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,8 @@ class QuadTree : public Shape {
2020

2121
float _totalX, _totalY;
2222
int _maxDepth;
23-
float precision;
24-
23+
float precision;
24+
int _area;
2525

2626
private:
2727
void copy(const QuadTree&);
@@ -33,17 +33,16 @@ class QuadTree : public Shape {
3333
~QuadTree();
3434

3535
public:
36-
QuadTree(Shape&, float precision);
37-
QuadTree(MultiPolygon& mult, float precision, unsigned id);
38-
39-
public:
40-
void translater(float x, float y);
41-
void rotater(float angle);
36+
QuadTree(Shape&, float precision);
37+
QuadTree(MultiPolygon& mult, float precision, unsigned id);
4238

4339
public:
40+
// Overrided algorithms
4441
void rotate(double degrees) override;
4542
void translate(double Tx, double Ty) override;
46-
bool intersectsWith(const QuadTree&) const;
43+
bool intersectsWith(const Shape&) const override;
44+
void envelope(Box&) const override;
45+
int area() const override;
4746

4847
private:
4948
void saveTree(std::string filename, int depth);

test/quadtreeTest.cpp

+67-5
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,8 @@ using Clock = std::chrono::system_clock;
2525
using namespace std;
2626

2727
int main() {
28+
static bool verboseDebug = false;
29+
2830
srand(time(0));
2931

3032
//Will contain the IDs the solver will pack
@@ -40,14 +42,15 @@ int main() {
4042
int celapsed = chrono::duration_cast<chrono::microseconds>(cend - cstart).count();
4143
cerr << "Parsing and quadtree creation time : " << celapsed << " microseconds elapsed" << endl;
4244

43-
44-
for(size_t i=0; i<shapes.size(); i++) {
45-
QuadTree& quad = dynamic_cast<QuadTree&>(shapes[i]);
46-
cout << quad << endl;
45+
if (verboseDebug) {
46+
for(size_t i=0; i<shapes.size(); i++) {
47+
QuadTree& quad = dynamic_cast<QuadTree&>(shapes[i]);
48+
cout << quad << endl;
49+
}
4750
}
4851

4952
// Intersection accuracy verification
50-
cerr << "Testting intersection accuracy :" << endl;
53+
cerr << "Testing intersection accuracy" << endl;
5154
bool noerr=true;
5255
for(size_t i=0; i<shapes.size(); i++) {
5356
for(size_t j=i+1; j<shapes.size(); j++) {
@@ -62,6 +65,65 @@ int main() {
6265
else
6366
cerr << "Some errors where find, see upside for more details" << endl;
6467

68+
69+
//Translation tests
70+
cerr << "Testing Translation accuracy" << endl;
71+
shapes[3].translate(-50, -150);
72+
ASSERT( shapes[0].intersectsWith(shapes[3]), "First Translation error 1");
73+
ASSERT( shapes[1].intersectsWith(shapes[3]), "First Translation error 2");
74+
ASSERT(!shapes[2].intersectsWith(shapes[3]), "First Translation error 3");
75+
ASSERT( shapes[4].intersectsWith(shapes[3]), "First Translation error 4");
76+
ASSERT( shapes[5].intersectsWith(shapes[3]), "First Translation error 5");
77+
78+
shapes[2].translate(50,-150);
79+
ASSERT(shapes[0].intersectsWith(shapes[2]), "Second Translation error 1");
80+
ASSERT(shapes[1].intersectsWith(shapes[2]), "Second Translation error 2");
81+
ASSERT(shapes[3].intersectsWith(shapes[2]), "Second Translation error 3");
82+
ASSERT(shapes[4].intersectsWith(shapes[2]), "Second Translation error 4");
83+
ASSERT(shapes[5].intersectsWith(shapes[2]), "Second Translation error 5");
84+
cerr << "No error found" << endl;
85+
86+
// Rotation test
87+
cerr << "Testing Rotation accuracy" << endl;
88+
Box test;
89+
shapes[0].rotate(90);
90+
shapes[0].envelope(test);
91+
ASSERT(abs(test.min_corner().x() - 0) < pow(10,-4), "Rotation 1-1");
92+
ASSERT(abs(test.min_corner().y() - -99.5) < pow(10,-4), "Rotation 1-2");
93+
ASSERT(abs(test.max_corner().x() - 99.5) < pow(10,-4), "Rotation 1-3");
94+
ASSERT(abs(test.max_corner().y() - 0) < pow(10,-4), "Rotation 1-4");
95+
if (verboseDebug)
96+
cout << dynamic_cast<QuadTree&>(shapes[0]) << endl;
97+
98+
shapes[0].rotate(90);
99+
shapes[0].envelope(test);
100+
ASSERT(abs(test.min_corner().x() - -99.5) < pow(10,-4), "Rotation 2-1");
101+
ASSERT(abs(test.min_corner().y() - -99.5) < pow(10,-4), "Rotation 2-2");
102+
ASSERT(abs(test.max_corner().x() - 0) < pow(10,-4), "Rotation 2-3");
103+
ASSERT(abs(test.max_corner().y() - 0) < pow(10,-4), "Rotation 2-4");
104+
if (verboseDebug)
105+
cout << dynamic_cast<QuadTree&>(shapes[0]) << endl;
106+
107+
shapes[0].rotate(90);
108+
shapes[0].envelope(test);
109+
ASSERT(abs(test.min_corner().x() - -99.5) < pow(10,-4), "Rotation 3-1");
110+
ASSERT(abs(test.min_corner().y() - 0) < pow(10,-4), "Rotation 3-2");
111+
ASSERT(abs(test.max_corner().x() - 0) < pow(10,-4), "Rotation 3-3");
112+
ASSERT(abs(test.max_corner().y() - 99.5) < pow(10,-4), "Rotation 3-4");
113+
if (verboseDebug)
114+
cout << dynamic_cast<QuadTree&>(shapes[0]) << endl;
115+
116+
shapes[0].rotate(90);
117+
shapes[0].envelope(test);
118+
ASSERT(abs(test.min_corner().x() - 0) < pow(10,-4), "Rotation 4-1");
119+
ASSERT(abs(test.min_corner().y() - 0) < pow(10,-4), "Rotation 4-2");
120+
ASSERT(abs(test.max_corner().x() - 99.5) < pow(10,-4), "Rotation 4-3");
121+
ASSERT(abs(test.max_corner().y() - 99.5) < pow(10,-4), "Rotation 4-4");
122+
if (verboseDebug)
123+
cout << dynamic_cast<QuadTree&>(shapes[0]) << endl;
124+
cerr << "No error found" << endl;
125+
126+
65127
/*bitmap bmap(shapes[1].getMultiP(), 99, 100);
66128
bmap.saveMap("test");
67129
for (size_t i=0; i<quads.size(); i++)

0 commit comments

Comments
 (0)