source: trunk/VUT/GtpVisibilityPreprocessor/src/Polygon3.cpp @ 256

Revision 256, 3.5 KB checked in by bittner, 19 years ago (diff)
Line 
1#include "Polygon3.h"
2#include "Mesh.h"
3
4#define SIDE_TOLERANCE 0.002 // TODO: Test different values
5
6Polygon3::Polygon3(): mMaterial(NULL)
7{}
8
9Polygon3::Polygon3(const VertexContainer &vertices): mVertices(vertices), mMaterial(NULL)
10{}
11
12Polygon3::Polygon3(Face *face, Mesh *parent)
13{
14        VertexIndexContainer::const_iterator it;
15
16        for (it = face->mVertexIndices.begin(); it != face->mVertexIndices.end(); ++ it)
17        {
18                mVertices.push_back(parent->mVertices[*it]);
19                mMaterial = parent->mMaterial;
20               
21                Debug << parent->mVertices[*it] << endl;
22        }
23}
24
25Plane3 Polygon3::GetSupportingPlane() const
26{
27        Vector3 v1 = mVertices[0] - mVertices[1];
28        Vector3 v2 = mVertices[2] - mVertices[1];
29#ifdef _DEBUG
30        Debug << "plane spanned by " <<  v1 << ", " << v2  << endl;
31#endif
32        return Plane3(mVertices[0], mVertices[1], mVertices[2]);
33}
34
35void Polygon3::DeletePolygons(PolygonContainer *polys)
36{
37        // don't need to store polygon information => delete polygons
38        while(!polys->empty())
39        {
40                Polygon3 *poly = polys->back();
41                polys->pop_back();
42                DEL_PTR(poly);
43        }
44}
45
46void Polygon3::Split(Plane3 *partition, Polygon3 *front, Polygon3 *back, int &splits)
47{
48        splits = 0;
49        Vector3 ptA = mVertices[mVertices.size() - 1];
50       
51        int sideA = partition->Side(ptA, SIDE_TOLERANCE);
52       
53        VertexContainer::const_iterator it;
54
55        // find line - plane intersections
56        for (it = mVertices.begin(); it != mVertices.end(); ++ it)
57        {
58                Vector3 ptB = (*it);
59                int sideB = partition->Side(ptB, SIDE_TOLERANCE);
60       
61                // vertices on different sides => split
62            if (sideB > 0)
63                {
64                        if (sideA < 0)
65                        {
66                                //-- plane - line intersection
67                                Vector3 splitPt = partition->FindIntersection(ptA, ptB);
68                       
69                                // add vertex to both polygons
70                                front->mVertices.push_back(splitPt);
71                                back->mVertices.push_back(splitPt);
72                       
73                                ++ splits;
74                        }
75                        front->mVertices.push_back(ptB);
76                }
77                else if (sideB < 0)
78                {
79                        if (sideA > 0)
80                        {
81                                //-- plane - line intersection
82                                Vector3 splitPt = partition->FindIntersection(ptA, ptB);
83                       
84                                // add vertex to both polygons
85                                front->mVertices.push_back(splitPt);
86                                back->mVertices.push_back(splitPt);
87
88                                ++ splits;
89                        }
90                        back->mVertices.push_back(ptB);
91                }
92                else
93                {
94                        // vertex on plane => add vertex to both polygons
95                        front->mVertices.push_back(ptB);
96                        back->mVertices.push_back(ptB);
97                }
98       
99                ptA = ptB;
100                sideA = sideB;
101        }
102}
103
104int Polygon3::Side(const Plane3 &plane) const
105{
106        int classification = ClassifyPlane(plane);
107       
108        if (classification == BACK_SIDE)
109                return -1;
110        else if (classification == FRONT_SIDE)
111                return 1;
112
113        return 0;
114}
115
116int Polygon3::ClassifyPlane(const Plane3 &plane) const
117{
118        VertexContainer::const_iterator it;
119
120        bool onFrontSide = false;
121        bool onBackSide = false;
122
123        // find possible line-plane intersections
124        for (it = mVertices.begin(); it != mVertices.end(); ++ it)
125        {
126                int side = plane.Side(*it, SIDE_TOLERANCE);
127               
128                if (side > 0)
129                {
130                        onFrontSide = true;
131                }
132                else if (side < 0)
133                {
134                        onBackSide = true;
135                }
136
137                if (onFrontSide && onBackSide) // split //TODO: check if through vvertex
138                {
139                        return SPLIT;
140                }
141        }
142
143        if (onBackSide)
144        {
145                return BACK_SIDE;
146        }
147        else if (onFrontSide)
148        {
149                return FRONT_SIDE;
150        }
151
152        // plane and polygon are coincident
153        return COINCIDENT;
154}
155
156
157Vector3
158Polygon3::Center() const
159{
160        int i;
161        Vector3 sum = mVertices[0];
162        for (i=1; i < mVertices.size(); i++)
163                sum += mVertices[i];
164       
165        return sum/i;
166}
167
168
169void
170Polygon3::Scale(const float scale)
171{
172        int i;
173        Vector3 center = Center();
174        for (i=0; i < mVertices.size(); i++) {
175                mVertices[i] = center + scale*(mVertices[i] - center);
176        }
177}
Note: See TracBrowser for help on using the repository browser.