1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #ifndef _BGFX_POLYGON_B2DPOLYGON_HXX
25 #define _BGFX_POLYGON_B2DPOLYGON_HXX
26 
27 #include <sal/types.h>
28 #include <o3tl/cow_wrapper.hxx>
29 #include <basegfx/vector/b2enums.hxx>
30 #include <basegfx/range/b2drange.hxx>
31 
32 //////////////////////////////////////////////////////////////////////////////
33 // predeclarations
34 class ImplB2DPolygon;
35 
36 namespace basegfx
37 {
38 	class B2DPolygon;
39 	class B2DPoint;
40 	class B2DVector;
41 	class B2DHomMatrix;
42     class B2DCubicBezier;
43 } // end of namespace basegfx
44 
45 //////////////////////////////////////////////////////////////////////////////
46 
47 namespace basegfx
48 {
49 	class B2DPolygon
50 	{
51     public:
52         typedef o3tl::cow_wrapper< ImplB2DPolygon > ImplType;
53 
54 	private:
55 		// internal data.
56         ImplType                                    mpPolygon;
57 
58 	public:
59 		/// diverse constructors
60 		B2DPolygon();
61 		B2DPolygon(const B2DPolygon& rPolygon);
62 		B2DPolygon(const B2DPolygon& rPolygon, sal_uInt32 nIndex, sal_uInt32 nCount);
63 		~B2DPolygon();
64 
65 		/// assignment operator
66 		B2DPolygon& operator=(const B2DPolygon& rPolygon);
67 
68         /// unshare this polygon with all internally shared instances
69         void makeUnique();
70 
71 		/// compare operators
72 		bool operator==(const B2DPolygon& rPolygon) const;
73 		bool operator!=(const B2DPolygon& rPolygon) const;
74 
75 		/// member count
76 		sal_uInt32 count() const;
77 
78 		/// Coordinate interface
79         basegfx::B2DPoint getB2DPoint(sal_uInt32 nIndex) const;
80 		void setB2DPoint(sal_uInt32 nIndex, const basegfx::B2DPoint& rValue);
81 
82 		/// Coordinate insert/append
83 		void insert(sal_uInt32 nIndex, const basegfx::B2DPoint& rPoint, sal_uInt32 nCount = 1);
84 		void append(const basegfx::B2DPoint& rPoint, sal_uInt32 nCount);
85 		void append(const basegfx::B2DPoint& rPoint);
86 		void reserve(sal_uInt32 nCount);
87 
88 		/// Basic ControlPoint interface
89 		basegfx::B2DPoint getPrevControlPoint(sal_uInt32 nIndex) const;
90 		basegfx::B2DPoint getNextControlPoint(sal_uInt32 nIndex) const;
91 		void setPrevControlPoint(sal_uInt32 nIndex, const basegfx::B2DPoint& rValue);
92 		void setNextControlPoint(sal_uInt32 nIndex, const basegfx::B2DPoint& rValue);
93 		void setControlPoints(sal_uInt32 nIndex, const basegfx::B2DPoint& rPrev, const basegfx::B2DPoint& rNext);
94 
95 		/// ControlPoint resets
96 		void resetPrevControlPoint(sal_uInt32 nIndex);
97 		void resetNextControlPoint(sal_uInt32 nIndex);
98 		void resetControlPoints(sal_uInt32 nIndex);
99 		void resetControlPoints();
100 
101 		/// Bezier segment append with control points. The current last polygon point is implicitly taken as start point.
102 		void appendBezierSegment(const basegfx::B2DPoint& rNextControlPoint, const basegfx::B2DPoint& rPrevControlPoint, const basegfx::B2DPoint& rPoint);
103 
104 		/// ControlPoint checks
105 		bool areControlPointsUsed() const;
106 		bool isPrevControlPointUsed(sal_uInt32 nIndex) const;
107 		bool isNextControlPointUsed(sal_uInt32 nIndex) const;
108 		B2VectorContinuity getContinuityInPoint(sal_uInt32 nIndex) const;
109 
110         /** check edge for being a bezier segment
111 
112             This test the existance of control vectors, but do not apply
113             testAndSolveTrivialBezier() to the bezier segment, so it is still useful
114             to do so.
115             Since it can use internal data representations, it is faster
116             than using getBezierSegment() and applying isBezier() on it.
117 
118             @param nIndex
119             Index of the addressed edge's start point
120 
121             @return
122             true if edge exists and at least one control vector is used
123         */
124         bool isBezierSegment(sal_uInt32 nIndex) const;
125 
126         /** bezier segment access
127 
128             This method also works when it is no bezier segment at all and will fill
129             the given B2DCubicBezier as needed.
130             In any case, the given B2DCubicBezier will be filled, if necessary with
131             the single start point (if no valid edge exists).
132 
133             @param nIndex
134             Index of the addressed edge's start point
135 
136             @param rTarget
137             The B2DCubicBezier to be filled. It's data WILL be changed.
138         */
139         void getBezierSegment(sal_uInt32 nIndex, B2DCubicBezier& rTarget) const;
140 
141 		/** Default adaptive subdivision access
142 
143 			This method will return a default adapive subdivision of the polygon.
144 			If the polygon does not contain any bezier curve segments, it will
145 			just return itself.
146 
147 			The subdivision is created on first request and buffered, so when using
148 			this subdivision You have the guarantee for fast accesses for multiple
149 			usages. It is intended for tooling usage for tasks which would be hard
150 			to accomplish on bezier segments (e.g. isInEpsilonRange).
151 
152 			The current default subdivision uses adaptiveSubdivideByCount with 9
153 			subdivisions which gives 10 edges and 11 points per segment and is
154 			usually pretty usable for processing purposes. There is no parameter
155 			passing here ATM but it may be changed on demand. If needed, a TYPE
156 			and PARAMETER (both defaulted) may be added to allow for switching
157 			between the different kinds of subdivisiond and passing them one
158 			parameter.
159 
160 			The lifetime of the buffered subdivision is based on polygon changes.
161 			When changing the polygon, it will be flushed. It is buffered at the
162 			refcounted implementation class, so it will survive copy by value and
163 			combinations in PolyPolygons.
164 
165 			@return
166 			The default (and buffered) subdivision of this polygon. It may
167 			be this polygon itself when it has no bezier segments. It is guaranteed
168 			to have no more bezier segments
169 		*/
170         B2DPolygon getDefaultAdaptiveSubdivision() const;
171 
172         /** Get the B2DRange (Rectangle dimensions) of this B2DPolygon
173 
174 			A polygon may have up to three ranges:
175 
176 			(a) the range of the polygon points
177 			(b) the range of the polygon points and control points
178 			(c) the outer range of the subdivided bezier curve
179 
180 			Ranges (a) and (c) are produced by tools::getRange(); resp. this
181             getB2DRange(). tools::getRangeWithControlPoints handles case (b).
182 
183 			To get range (c) a simple solution would be to subdivide the polygon
184 			and use getRange() on it. Since subdivision is expensive and decreases
185 			the polygon quality, i added this new method. It will use a
186 			methodology suggested by HDU. First, it gets the range (a).
187 			Then it iterates over the bezier segments and for each it
188 			first tests if the outer range of the bezier segment is already
189 			contained in the result range.
190 
191 			The subdivision itself uses getAllExtremumPositions() to only
192 			calculate extremum points and to expand the result accordingly.
193 			Thus it calculates maximal four extremum points on the bezier
194 			segment, no split is used at all.
195 
196 			@return
197 			The outer range of the bezier curve/polygon
198         */
199         B2DRange getB2DRange() const;
200 
201 		/** insert other 2D polygons
202 
203 			The default (with nIndex2 == 0 && nCount == 0) inserts the whole
204 			rPoly at position nIndex
205 
206 			@param nIndex
207 			Target index for points to be inserted
208 
209 			@param rPoly
210 			The source for new points
211 
212 			@param nIndex2
213 			The index to the first source point into rPoly
214 
215 			@param nCount
216 			How many points to add from rPoly to this polygon. Null
217 			means to copy all (starting from nIndex2)
218 		*/
219 		void insert(sal_uInt32 nIndex, const B2DPolygon& rPoly, sal_uInt32 nIndex2 = 0, sal_uInt32 nCount = 0);
220 
221 		/** append other 2D polygons
222 
223 			The default (nIndex ==0 && nCount == 0) will append
224 			the whole rPoly
225 
226 			@param rPoly
227 			The source polygon
228 
229 			@param nIndex
230 			The index to the first point of rPoly to append
231 
232 			@param nCount
233 			The number of points to append from rPoly, starting
234 			from nIndex. If zero, as much as possibel is appended
235 		*/
236 		void append(const B2DPolygon& rPoly, sal_uInt32 nIndex = 0, sal_uInt32 nCount = 0);
237 
238 		/// remove points
239 		void remove(sal_uInt32 nIndex, sal_uInt32 nCount = 1);
240 
241 		/// clear all points
242 		void clear();
243 
244 		/// closed state interface
245 		bool isClosed() const;
246 		void setClosed(bool bNew);
247 
248 		/// flip polygon direction
249 		void flip();
250 
251 		/// test if Polygon has double points
252 		bool hasDoublePoints() const;
253 
254 		/// remove double points, at the begin/end and follow-ups, too
255 		void removeDoublePoints();
256 
257 		/// apply transformation given in matrix form
258 		void transform(const basegfx::B2DHomMatrix& rMatrix);
259 
260         // point iterators (same iterator validity conditions as for vector)
261         const B2DPoint* begin() const;
262         const B2DPoint* end() const;
263         B2DPoint* begin();
264         B2DPoint* end();
265 	};
266 } // end of namespace basegfx
267 
268 //////////////////////////////////////////////////////////////////////////////
269 
270 #endif /* _BGFX_POLYGON_B2DPOLYGON_HXX */
271