Code

2geom update (rev. 1578); fixes node editing of some degenerate paths
[inkscape.git] / src / 2geom / path.cpp
1 /*
2  * Path - Series of continuous curves
3  *
4  * Authors:
5  *              MenTaLguY <mental@rydia.net>
6  *              Marco Cecchetti <mrcekets at gmail.com>
7  *
8  * Copyright 2007-2008  authors
9  *
10  * This library is free software; you can redistribute it and/or
11  * modify it either under the terms of the GNU Lesser General Public
12  * License version 2.1 as published by the Free Software Foundation
13  * (the "LGPL") or, at your option, under the terms of the Mozilla
14  * Public License Version 1.1 (the "MPL"). If you do not alter this
15  * notice, a recipient may use your version of this file under either
16  * the MPL or the LGPL.
17  *
18  * You should have received a copy of the LGPL along with this library
19  * in the file COPYING-LGPL-2.1; if not, write to the Free Software
20  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21  * You should have received a copy of the MPL along with this library
22  * in the file COPYING-MPL-1.1
23  *
24  * The contents of this file are subject to the Mozilla Public License
25  * Version 1.1 (the "License"); you may not use this file except in
26  * compliance with the License. You may obtain a copy of the License at
27  * http://www.mozilla.org/MPL/
28  *
29  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY
30  * OF ANY KIND, either express or implied. See the LGPL or the MPL for
31  * the specific language governing rights and limitations.
32  */
37 #include <2geom/path.h>
38 #include <algorithm>
41 using namespace Geom::PathInternal;
43 namespace Geom
44 {
46 Rect Path::boundsFast() const {
47   Rect bounds;
48   if (empty()) return bounds;
49   bounds=front().boundsFast();
50   const_iterator iter = begin();
51   if ( iter != end() ) {
52           for ( ++iter; iter != end() ; ++iter ) {
53             bounds.unionWith(iter->boundsFast());
54           }
55   }
56   return bounds;
57 }
59 Rect Path::boundsExact() const {
60   Rect bounds;
61   if (empty()) return bounds;
62   bounds=front().boundsExact();
63   const_iterator iter = begin();
64   if ( iter != end() ) {
65     for ( ++iter; iter != end() ; ++iter ) {
66       bounds.unionWith(iter->boundsExact());
67     }
68   }
69   return bounds;
70 }
72 template<typename iter>
73 iter inc(iter const &x, unsigned n) {
74   iter ret = x;
75   for(unsigned i = 0; i < n; i++)
76     ret++;
77   return ret;
78 }
80 Path &Path::operator*=(Matrix const &m) {
81   unshare();
82   Sequence::iterator last = get_curves().end() - 1;
83   Sequence::iterator it;
84   Point prev;
85   for (it = get_curves().begin() ; it != last ; ++it) {
86     *it = boost::shared_ptr<Curve>((*it)->transformed(m));
87     if ( it != get_curves().begin() && (*it)->initialPoint() != prev ) {
88       THROW_CONTINUITYERROR();
89     }
90     prev = (*it)->finalPoint();
91   }
92   for ( int i = 0 ; i < 2 ; ++i ) {
93     final_->setPoint(i, (*final_)[i] * m);
94   }
95   if (get_curves().size() > 1) {
96     if ( front().initialPoint() != initialPoint() || back().finalPoint() != finalPoint() ) {
97       THROW_CONTINUITYERROR();
98     }
99   }
100   return *this;
103 std::vector<double>
104 Path::allNearestPoints(Point const& _point, double from, double to) const
106         if ( from > to ) std::swap(from, to);
107         const Path& _path = *this;
108         unsigned int sz = _path.size();
109         if ( _path.closed() ) ++sz;
110         if ( from < 0 || to > sz )
111         {
112                 THROW_RANGEERROR("[from,to] interval out of bounds");
113         }
114         double sif, st = modf(from, &sif);
115         double eif, et = modf(to, &eif);
116         unsigned int si = static_cast<unsigned int>(sif);
117         unsigned int ei = static_cast<unsigned int>(eif);
118         if ( si == sz )
119         {
120                 --si;
121                 st = 1;
122         }
123         if ( ei == sz )
124         {
125                 --ei;
126                 et = 1;
127         }
128         if ( si == ei )
129         {
130                 std::vector<double>     all_nearest =
131                         _path[si].allNearestPoints(_point, st, et);
132                 for ( unsigned int i = 0; i < all_nearest.size(); ++i )
133                 {
134                         all_nearest[i] = si + all_nearest[i];
135                 }
136                 return all_nearest;
137         }
138         std::vector<double> all_t;
139         std::vector< std::vector<double> > all_np;
140         all_np.push_back( _path[si].allNearestPoints(_point, st) );
141         std::vector<unsigned int> ni;
142         ni.push_back(si);
143         double dsq;
144         double mindistsq
145                 = distanceSq( _point, _path[si].pointAt( all_np.front().front() ) );
146         Rect bb;
147         for ( unsigned int i = si + 1; i < ei; ++i )
148         {
149                 bb = _path[i].boundsFast();
150                 dsq = distanceSq(_point, bb);
151                 if ( mindistsq < dsq ) continue;
152                 all_t = _path[i].allNearestPoints(_point);
153                 dsq = distanceSq( _point, _path[i].pointAt( all_t.front() ) );
154                 if ( mindistsq > dsq )
155                 {
156                         all_np.clear();
157                         all_np.push_back(all_t);
158                         ni.clear();
159                         ni.push_back(i);
160                         mindistsq = dsq;
161                 }
162                 else if ( mindistsq == dsq )
163                 {
164                         all_np.push_back(all_t);
165                         ni.push_back(i);
166                 }
167         }
168         bb = _path[ei].boundsFast();
169         dsq = distanceSq(_point, bb);
170         if ( mindistsq >= dsq )
171         {
172                 all_t = _path[ei].allNearestPoints(_point, 0, et);
173                 dsq = distanceSq( _point, _path[ei].pointAt( all_t.front() ) );
174                 if ( mindistsq > dsq )
175                 {
176                         for ( unsigned int i = 0; i < all_t.size(); ++i )
177                         {
178                                 all_t[i] = ei + all_t[i];
179                         }
180                         return all_t;
181                 }
182                 else if ( mindistsq == dsq )
183                 {
184                         all_np.push_back(all_t);
185                         ni.push_back(ei);
186                 }
187         }
188         std::vector<double> all_nearest;
189         for ( unsigned int i = 0; i < all_np.size(); ++i )
190         {
191                 for ( unsigned int j = 0; j < all_np[i].size(); ++j )
192                 {
193                         all_nearest.push_back( ni[i] + all_np[i][j] );
194                 }
195         }
196         all_nearest.erase(std::unique(all_nearest.begin(), all_nearest.end()),
197                           all_nearest.end());
198         return all_nearest;
201 std::vector<double>
202 Path::nearestPointPerCurve(Point const& _point) const
204         //return a single nearest point for each curve in this path
205         std::vector<double> np;
206         const Path& _path = *this;
207         for (Sequence::const_iterator it = _path.get_curves().begin() ; it != _path.get_curves().end()-1 ; ++it)
208         //for (std::vector<Path>::const_iterator it = _path.begin(); it != _path.end(), ++it){
209         {
210             np.push_back((*it)->nearestPoint(_point));
211     }
212         return np;
213 }  
215 double Path::nearestPoint(Point const &_point, double from, double to, double *distance_squared) const
217         if ( from > to ) std::swap(from, to);
218         const Path& _path = *this;
219         unsigned int sz = _path.size();
220         if ( _path.closed() ) ++sz;
221         if ( from < 0 || to > sz )
222         {
223                 THROW_RANGEERROR("[from,to] interval out of bounds");
224         }
225         double sif, st = modf(from, &sif);
226         double eif, et = modf(to, &eif);
227         unsigned int si = static_cast<unsigned int>(sif);
228         unsigned int ei = static_cast<unsigned int>(eif);
229         if(sz == 0) {// naked moveto
230             if (distance_squared != NULL)
231                 *distance_squared = distanceSq(_point, _path.initialPoint());
232             return 0;
233         }
234         if ( si == sz )
235         {
236                 --si;
237                 st = 1;
238         }
239         if ( ei == sz )
240         {
241                 --ei;
242                 et = 1;
243         }
244         if ( si == ei )
245         {
246                 double nearest = _path[si].nearestPoint(_point, st, et);
247                 if (distance_squared != NULL)
248                     *distance_squared = distanceSq(_point, _path[si].pointAt(nearest));
249                 return si + nearest;
250         }
252         double t;
253         double nearest = _path[si].nearestPoint(_point, st);
254         unsigned int ni = si;
255         double dsq;
256         double mindistsq = distanceSq(_point, _path[si].pointAt(nearest));
257         Rect bb;
258         for ( unsigned int i = si + 1; i < ei; ++i )
259         {
260                 bb = _path[i].boundsFast();
261                 dsq = distanceSq(_point, bb);
262                 if ( mindistsq <= dsq ) continue;
263                 t = _path[i].nearestPoint(_point);
264                 dsq = distanceSq(_point, _path[i].pointAt(t));
265                 if ( mindistsq > dsq )
266                 {
267                         nearest = t;
268                         ni = i;
269                         mindistsq = dsq;
270                 }
271         }
272         bb = _path[ei].boundsFast();
273         dsq = distanceSq(_point, bb);
274         if ( mindistsq > dsq )
275         {
276                 t = _path[ei].nearestPoint(_point, 0, et);
277                 dsq = distanceSq(_point, _path[ei].pointAt(t));
278                 if ( mindistsq > dsq )
279                 {
280                         nearest = t;
281                         ni = ei;
282                         mindistsq = dsq;
283                 }
284         }
286     if (distance_squared != NULL)
287         *distance_squared = mindistsq;
289         return ni + nearest;
292 void Path::appendPortionTo(Path &ret, double from, double to) const {
293   assert(from >= 0 && to >= 0);
294   if(to == 0) to = size()+0.999999;
295   if(from == to) { return; }
296   double fi, ti;
297   double ff = modf(from, &fi), tf = modf(to, &ti);
298   if(tf == 0) { ti--; tf = 1; }
299   const_iterator fromi = inc(begin(), (unsigned)fi);
300   if(fi == ti && from < to) {
301     Curve *v = fromi->portion(ff, tf);
302     ret.append(*v, STITCH_DISCONTINUOUS);
303     delete v;
304     return;
305   }
306   const_iterator toi = inc(begin(), (unsigned)ti);
307   if(ff != 1.) {
308     Curve *fromv = fromi->portion(ff, 1.);
309     //fromv->setInitial(ret.finalPoint());
310     ret.append(*fromv, STITCH_DISCONTINUOUS);
311     delete fromv;
312   }
313   if(from >= to) {
314     const_iterator ender = end();
315     if(ender->initialPoint() == ender->finalPoint()) ender++;
316     ret.insert(ret.end(), ++fromi, ender, STITCH_DISCONTINUOUS);
317     ret.insert(ret.end(), begin(), toi, STITCH_DISCONTINUOUS);
318   } else {
319     ret.insert(ret.end(), ++fromi, toi, STITCH_DISCONTINUOUS);
320   }
321   Curve *tov = toi->portion(0., tf);
322   ret.append(*tov, STITCH_DISCONTINUOUS);
323   delete tov;
326 void Path::do_update(Sequence::iterator first_replaced,
327                      Sequence::iterator last_replaced,
328                      Sequence::iterator first,
329                      Sequence::iterator last)
331   // note: modifies the contents of [first,last)
332   check_continuity(first_replaced, last_replaced, first, last);
333   if ( ( last - first ) == ( last_replaced - first_replaced ) ) {
334     std::copy(first, last, first_replaced);
335   } else {
336     // this approach depends on std::vector's behavior WRT iterator stability
337     get_curves().erase(first_replaced, last_replaced);
338     get_curves().insert(first_replaced, first, last);
339   }
341   if ( get_curves().front().get() != final_ ) {
342     final_->setPoint(0, back().finalPoint());
343     final_->setPoint(1, front().initialPoint());
344   }
347 void Path::do_append(Curve *c) {
348   boost::shared_ptr<Curve> curve(c);
349   if ( get_curves().front().get() == final_ ) {
350     final_->setPoint(1, curve->initialPoint());
351   } else {
352     if (curve->initialPoint() != finalPoint()) {
353       THROW_CONTINUITYERROR();
354     }
355   }
356   get_curves().insert(get_curves().end()-1, curve);
357   final_->setPoint(0, curve->finalPoint());
360 void Path::stitch(Sequence::iterator first_replaced,
361                   Sequence::iterator last_replaced,
362                   Sequence &source)
364   if (!source.empty()) {
365     if ( first_replaced != get_curves().begin() ) {
366       if ( (*first_replaced)->initialPoint() != source.front()->initialPoint() ) {
367         Curve *stitch = new StitchSegment((*first_replaced)->initialPoint(),
368                                           source.front()->initialPoint());
369         source.insert(source.begin(), boost::shared_ptr<Curve>(stitch));
370       }
371     }
372     if ( last_replaced != (get_curves().end()-1) ) {
373       if ( (*last_replaced)->finalPoint() != source.back()->finalPoint() ) {
374         Curve *stitch = new StitchSegment(source.back()->finalPoint(),
375                                           (*last_replaced)->finalPoint());
376         source.insert(source.end(), boost::shared_ptr<Curve>(stitch));
377       }
378     }
379   } else if ( first_replaced != last_replaced && first_replaced != get_curves().begin() && last_replaced != get_curves().end()-1) {
380     if ( (*first_replaced)->initialPoint() != (*(last_replaced-1))->finalPoint() ) {
381       Curve *stitch = new StitchSegment((*(last_replaced-1))->finalPoint(),
382                                         (*first_replaced)->initialPoint());
383       source.insert(source.begin(), boost::shared_ptr<Curve>(stitch));
384     }
385   }
388 void Path::check_continuity(Sequence::iterator first_replaced,
389                             Sequence::iterator last_replaced,
390                             Sequence::iterator first,
391                             Sequence::iterator last)
393   if ( first != last ) {
394     if ( first_replaced != get_curves().begin() ) {
395       if ( (*first_replaced)->initialPoint() != (*first)->initialPoint() ) {
396         THROW_CONTINUITYERROR();
397       }
398     }
399     if ( last_replaced != (get_curves().end()-1) ) {
400       if ( (*(last_replaced-1))->finalPoint() != (*(last-1))->finalPoint() ) {
401         THROW_CONTINUITYERROR();
402       }
403     }
404   } else if ( first_replaced != last_replaced && first_replaced != get_curves().begin() && last_replaced != get_curves().end()-1) {
405     if ( (*first_replaced)->initialPoint() != (*(last_replaced-1))->finalPoint() ) {
406       THROW_CONTINUITYERROR();
407     }
408   }
411 } // end namespace Geom
413 /*
414   Local Variables:
415   mode:c++
416   c-file-style:"stroustrup"
417   c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
418   indent-tabs-mode:nil
419   fill-column:99
420   End:
421 */
422 // vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99 :