1 /*M///////////////////////////////////////////////////////////////////////////////////////
2 //
3 //  IMPORTANT: READ BEFORE DOWNLOADING, COPYING, INSTALLING OR USING.
4 //
5 //  By downloading, copying, installing or using the software you agree to this license.
6 //  If you do not agree to this license, do not download, install,
7 //  copy or use the software.
8 //
9 //
10 //                        Intel License Agreement
11 //                For Open Source Computer Vision Library
12 //
13 // Copyright (C) 2000, Intel Corporation, all rights reserved.
14 // Third party copyrights are property of their respective owners.
15 //
16 // Redistribution and use in source and binary forms, with or without modification,
17 // are permitted provided that the following conditions are met:
18 //
19 //   * Redistribution's of source code must retain the above copyright notice,
20 //     this list of conditions and the following disclaimer.
21 //
22 //   * Redistribution's in binary form must reproduce the above copyright notice,
23 //     this list of conditions and the following disclaimer in the documentation
24 //     and/or other materials provided with the distribution.
25 //
26 //   * The name of Intel Corporation may not be used to endorse or promote products
27 //     derived from this software without specific prior written permission.
28 //
29 // This software is provided by the copyright holders and contributors "as is" and
30 // any express or implied warranties, including, but not limited to, the implied
31 // warranties of merchantability and fitness for a particular purpose are disclaimed.
32 // In no event shall the Intel Corporation or contributors be liable for any direct,
33 // indirect, incidental, special, exemplary, or consequential damages
34 // (including, but not limited to, procurement of substitute goods or services;
35 // loss of use, data, or profits; or business interruption) however caused
36 // and on any theory of liability, whether in contract, strict liability,
37 // or tort (including negligence or otherwise) arising in any way out of
38 // the use of this software, even if advised of the possibility of such damage.
39 //
40 //M*/
41 #include "_cxcore.h"
42 
43 #define  XY_SHIFT  16
44 #define  XY_ONE    (1 << XY_SHIFT)
45 
46 #define CV_DRAWING_STORAGE_BLOCK ((1 << 12) - 256)
47 
48 typedef struct CvPolyEdge
49 {
50     int x, dx;
51     union
52     {
53         struct CvPolyEdge *next;
54         int y0;
55     };
56     int y1;
57 }
58 CvPolyEdge;
59 
60 static void
61 icvCollectPolyEdges( CvMat* img, CvSeq* v, CvContour* edges,
62                      const void* color, int line_type,
63                      int shift, CvPoint offset=cvPoint(0,0) );
64 
65 static void
66 icvFillEdgeCollection( CvMat* img, CvContour* edges, const void* color );
67 
68 static void
69 icvPolyLine( CvMat* img, CvPoint *v, int count, int closed,
70              const void* color, int thickness, int line_type, int shift );
71 
72 static void
73 icvFillConvexPoly( CvMat* img, CvPoint* v, int npts,
74                    const void* color, int line_type, int shift );
75 
76 /****************************************************************************************\
77 *                                   Lines                                                *
78 \****************************************************************************************/
79 
80 CV_IMPL int
cvClipLine(CvSize img_size,CvPoint * pt1,CvPoint * pt2)81 cvClipLine( CvSize img_size, CvPoint* pt1, CvPoint* pt2 )
82 {
83     int result = 0;
84 
85     CV_FUNCNAME( "cvClipLine" );
86 
87     __BEGIN__;
88 
89     int x1, y1, x2, y2;
90     int c1, c2;
91     int right = img_size.width-1, bottom = img_size.height-1;
92 
93     if( !pt1 || !pt2 )
94         CV_ERROR( CV_StsNullPtr, "One of point pointers is NULL" );
95 
96     if( right < 0 || bottom < 0 )
97         CV_ERROR( CV_StsOutOfRange, "Image width or height are negative" );
98 
99     x1 = pt1->x; y1 = pt1->y; x2 = pt2->x; y2 = pt2->y;
100     c1 = (x1 < 0) + (x1 > right) * 2 + (y1 < 0) * 4 + (y1 > bottom) * 8;
101     c2 = (x2 < 0) + (x2 > right) * 2 + (y2 < 0) * 4 + (y2 > bottom) * 8;
102 
103     if( (c1 & c2) == 0 && (c1 | c2) != 0 )
104     {
105         int a;
106 
107         if( c1 & 12 )
108         {
109             a = c1 < 8 ? 0 : bottom;
110             x1 += (int) (((int64) (a - y1)) * (x2 - x1) / (y2 - y1));
111             y1 = a;
112             c1 = (x1 < 0) + (x1 > right) * 2;
113         }
114         if( c2 & 12 )
115         {
116             a = c2 < 8 ? 0 : bottom;
117             x2 += (int) (((int64) (a - y2)) * (x2 - x1) / (y2 - y1));
118             y2 = a;
119             c2 = (x2 < 0) + (x2 > right) * 2;
120         }
121         if( (c1 & c2) == 0 && (c1 | c2) != 0 )
122         {
123             if( c1 )
124             {
125                 a = c1 == 1 ? 0 : right;
126                 y1 += (int) (((int64) (a - x1)) * (y2 - y1) / (x2 - x1));
127                 x1 = a;
128                 c1 = 0;
129             }
130             if( c2 )
131             {
132                 a = c2 == 1 ? 0 : right;
133                 y2 += (int) (((int64) (a - x2)) * (y2 - y1) / (x2 - x1));
134                 x2 = a;
135                 c2 = 0;
136             }
137         }
138 
139         assert( (c1 & c2) != 0 || (x1 | y1 | x2 | y2) >= 0 );
140 
141         pt1->x = x1;
142         pt1->y = y1;
143         pt2->x = x2;
144         pt2->y = y2;
145     }
146 
147     result = ( c1 | c2 ) == 0;
148 
149     __END__;
150 
151     return result;
152 }
153 
154 
155 /*
156    Initializes line iterator.
157    Returns number of points on the line or negative number if error.
158 */
159 CV_IMPL int
cvInitLineIterator(const CvArr * img,CvPoint pt1,CvPoint pt2,CvLineIterator * iterator,int connectivity,int left_to_right)160 cvInitLineIterator( const CvArr* img, CvPoint pt1, CvPoint pt2,
161                     CvLineIterator* iterator, int connectivity,
162                     int left_to_right )
163 {
164     int count = -1;
165 
166     CV_FUNCNAME( "cvInitLineIterator" );
167 
168     __BEGIN__;
169 
170     CvMat stub, *mat = (CvMat*)img;
171     int dx, dy, s;
172     int bt_pix, bt_pix0, step;
173 
174     if( !CV_IS_MAT(mat) )
175         CV_CALL( mat = cvGetMat( mat, &stub ));
176 
177     if( !iterator )
178         CV_ERROR( CV_StsNullPtr, "Pointer to the iterator state is NULL" );
179 
180     if( connectivity != 8 && connectivity != 4 )
181         CV_ERROR( CV_StsBadArg, "Connectivity must be 8 or 4" );
182 
183     if( (unsigned)pt1.x >= (unsigned)(mat->width) ||
184         (unsigned)pt2.x >= (unsigned)(mat->width) ||
185         (unsigned)pt1.y >= (unsigned)(mat->height) ||
186         (unsigned)pt2.y >= (unsigned)(mat->height) )
187         CV_ERROR( CV_StsBadPoint,
188             "One of the ending points is outside of the image, use cvClipLine" );
189 
190     bt_pix0 = bt_pix = CV_ELEM_SIZE(mat->type);
191     step = mat->step;
192 
193     dx = pt2.x - pt1.x;
194     dy = pt2.y - pt1.y;
195     s = dx < 0 ? -1 : 0;
196 
197     if( left_to_right )
198     {
199         dx = (dx ^ s) - s;
200         dy = (dy ^ s) - s;
201         pt1.x ^= (pt1.x ^ pt2.x) & s;
202         pt1.y ^= (pt1.y ^ pt2.y) & s;
203     }
204     else
205     {
206         dx = (dx ^ s) - s;
207         bt_pix = (bt_pix ^ s) - s;
208     }
209 
210     iterator->ptr = (uchar*)(mat->data.ptr + pt1.y * step + pt1.x * bt_pix0);
211 
212     s = dy < 0 ? -1 : 0;
213     dy = (dy ^ s) - s;
214     step = (step ^ s) - s;
215 
216     s = dy > dx ? -1 : 0;
217 
218     /* conditional swaps */
219     dx ^= dy & s;
220     dy ^= dx & s;
221     dx ^= dy & s;
222 
223     bt_pix ^= step & s;
224     step ^= bt_pix & s;
225     bt_pix ^= step & s;
226 
227     if( connectivity == 8 )
228     {
229         assert( dx >= 0 && dy >= 0 );
230 
231         iterator->err = dx - (dy + dy);
232         iterator->plus_delta = dx + dx;
233         iterator->minus_delta = -(dy + dy);
234         iterator->plus_step = step;
235         iterator->minus_step = bt_pix;
236         count = dx + 1;
237     }
238     else /* connectivity == 4 */
239     {
240         assert( dx >= 0 && dy >= 0 );
241 
242         iterator->err = 0;
243         iterator->plus_delta = (dx + dx) + (dy + dy);
244         iterator->minus_delta = -(dy + dy);
245         iterator->plus_step = step - bt_pix;
246         iterator->minus_step = bt_pix;
247         count = dx + dy + 1;
248     }
249 
250     __END__;
251 
252     return count;
253 }
254 
255 static void
icvLine(CvMat * mat,CvPoint pt1,CvPoint pt2,const void * color,int connectivity=8)256 icvLine( CvMat* mat, CvPoint pt1, CvPoint pt2,
257          const void* color, int connectivity = 8 )
258 {
259     if( cvClipLine( cvGetMatSize(mat), &pt1, &pt2 ))
260     {
261         CvLineIterator iterator;
262         int pix_size = CV_ELEM_SIZE(mat->type);
263         int i, count;
264 
265         if( connectivity == 0 )
266             connectivity = 8;
267         if( connectivity == 1 )
268             connectivity = 4;
269 
270         count = cvInitLineIterator( mat, pt1, pt2, &iterator, connectivity, 1 );
271 
272         for( i = 0; i < count; i++ )
273         {
274             CV_MEMCPY_AUTO( iterator.ptr, color, pix_size );
275             CV_NEXT_LINE_POINT( iterator );
276         }
277     }
278 }
279 
280 
281 /* Correction table depent on the slope */
282 static const uchar icvSlopeCorrTable[] = {
283     181, 181, 181, 182, 182, 183, 184, 185, 187, 188, 190, 192, 194, 196, 198, 201,
284     203, 206, 209, 211, 214, 218, 221, 224, 227, 231, 235, 238, 242, 246, 250, 254
285 };
286 
287 /* Gaussian for antialiasing filter */
288 static const int icvFilterTable[] = {
289     168, 177, 185, 194, 202, 210, 218, 224, 231, 236, 241, 246, 249, 252, 254, 254,
290     254, 254, 252, 249, 246, 241, 236, 231, 224, 218, 210, 202, 194, 185, 177, 168,
291     158, 149, 140, 131, 122, 114, 105, 97, 89, 82, 75, 68, 62, 56, 50, 45,
292     40, 36, 32, 28, 25, 22, 19, 16, 14, 12, 11, 9, 8, 7, 5, 5
293 };
294 
295 static void
icvLineAA(CvMat * img,CvPoint pt1,CvPoint pt2,const void * color)296 icvLineAA( CvMat* img, CvPoint pt1, CvPoint pt2,
297            const void* color )
298 {
299     int dx, dy;
300     int ecount, scount = 0;
301     int slope;
302     int ax, ay;
303     int x_step, y_step;
304     int i, j;
305     int ep_table[9];
306     int cb = ((uchar*)color)[0], cg = ((uchar*)color)[1], cr = ((uchar*)color)[2];
307     int _cb, _cg, _cr;
308     int nch = CV_MAT_CN( img->type );
309     uchar* ptr = (uchar*)(img->data.ptr);
310     int step = img->step;
311     CvSize size = cvGetMatSize( img );
312 
313     assert( img && (nch == 1 || nch == 3) && CV_MAT_DEPTH(img->type) == CV_8U );
314 
315     pt1.x -= XY_ONE*2;
316     pt1.y -= XY_ONE*2;
317     pt2.x -= XY_ONE*2;
318     pt2.y -= XY_ONE*2;
319     ptr += img->step*2 + 2*nch;
320 
321     size.width = ((size.width - 5) << XY_SHIFT) + 1;
322     size.height = ((size.height - 5) << XY_SHIFT) + 1;
323 
324     if( !cvClipLine( size, &pt1, &pt2 ))
325         return;
326 
327     dx = pt2.x - pt1.x;
328     dy = pt2.y - pt1.y;
329 
330     j = dx < 0 ? -1 : 0;
331     ax = (dx ^ j) - j;
332     i = dy < 0 ? -1 : 0;
333     ay = (dy ^ i) - i;
334 
335     if( ax > ay )
336     {
337         dx = ax;
338         dy = (dy ^ j) - j;
339         pt1.x ^= pt2.x & j;
340         pt2.x ^= pt1.x & j;
341         pt1.x ^= pt2.x & j;
342         pt1.y ^= pt2.y & j;
343         pt2.y ^= pt1.y & j;
344         pt1.y ^= pt2.y & j;
345 
346         x_step = XY_ONE;
347         y_step = (int) (((int64) dy << XY_SHIFT) / (ax | 1));
348         pt2.x += XY_ONE;
349         ecount = (pt2.x >> XY_SHIFT) - (pt1.x >> XY_SHIFT);
350         j = -(pt1.x & (XY_ONE - 1));
351         pt1.y += (int) ((((int64) y_step) * j) >> XY_SHIFT) + (XY_ONE >> 1);
352         slope = (y_step >> (XY_SHIFT - 5)) & 0x3f;
353         slope ^= (y_step < 0 ? 0x3f : 0);
354 
355         /* Get 4-bit fractions for end-point adjustments */
356         i = (pt1.x >> (XY_SHIFT - 7)) & 0x78;
357         j = (pt2.x >> (XY_SHIFT - 7)) & 0x78;
358     }
359     else
360     {
361         dy = ay;
362         dx = (dx ^ i) - i;
363         pt1.x ^= pt2.x & i;
364         pt2.x ^= pt1.x & i;
365         pt1.x ^= pt2.x & i;
366         pt1.y ^= pt2.y & i;
367         pt2.y ^= pt1.y & i;
368         pt1.y ^= pt2.y & i;
369 
370         x_step = (int) (((int64) dx << XY_SHIFT) / (ay | 1));
371         y_step = XY_ONE;
372         pt2.y += XY_ONE;
373         ecount = (pt2.y >> XY_SHIFT) - (pt1.y >> XY_SHIFT);
374         j = -(pt1.y & (XY_ONE - 1));
375         pt1.x += (int) ((((int64) x_step) * j) >> XY_SHIFT) + (XY_ONE >> 1);
376         slope = (x_step >> (XY_SHIFT - 5)) & 0x3f;
377         slope ^= (x_step < 0 ? 0x3f : 0);
378 
379         /* Get 4-bit fractions for end-point adjustments */
380         i = (pt1.y >> (XY_SHIFT - 7)) & 0x78;
381         j = (pt2.y >> (XY_SHIFT - 7)) & 0x78;
382     }
383 
384     slope = (slope & 0x20) ? 0x100 : icvSlopeCorrTable[slope];
385 
386     /* Calc end point correction table */
387     {
388         int t0 = slope << 7;
389         int t1 = ((0x78 - i) | 4) * slope;
390         int t2 = (j | 4) * slope;
391 
392         ep_table[0] = 0;
393         ep_table[8] = slope;
394         ep_table[1] = ep_table[3] = ((((j - i) & 0x78) | 4) * slope >> 8) & 0x1ff;
395         ep_table[2] = (t1 >> 8) & 0x1ff;
396         ep_table[4] = ((((j - i) + 0x80) | 4) * slope >> 8) & 0x1ff;
397         ep_table[5] = ((t1 + t0) >> 8) & 0x1ff;
398         ep_table[6] = (t2 >> 8) & 0x1ff;
399         ep_table[7] = ((t2 + t0) >> 8) & 0x1ff;
400     }
401 
402     if( nch == 3 )
403     {
404         #define  ICV_PUT_POINT()            \
405         {                                   \
406             _cb = tptr[0];                  \
407             _cb += ((cb - _cb)*a + 127)>> 8;\
408             _cg = tptr[1];                  \
409             _cg += ((cg - _cg)*a + 127)>> 8;\
410             _cr = tptr[2];                  \
411             _cr += ((cr - _cr)*a + 127)>> 8;\
412             tptr[0] = (uchar)_cb;           \
413             tptr[1] = (uchar)_cg;           \
414             tptr[2] = (uchar)_cr;           \
415         }
416         if( ax > ay )
417         {
418             ptr += (pt1.x >> XY_SHIFT) * 3;
419 
420             while( ecount >= 0 )
421             {
422                 uchar *tptr = ptr + ((pt1.y >> XY_SHIFT) - 1) * step;
423 
424                 int ep_corr = ep_table[(((scount >= 2) + 1) & (scount | 2)) * 3 +
425                                        (((ecount >= 2) + 1) & (ecount | 2))];
426                 int a, dist = (pt1.y >> (XY_SHIFT - 5)) & 31;
427 
428                 a = (ep_corr * icvFilterTable[dist + 32] >> 8) & 0xff;
429                 ICV_PUT_POINT();
430                 ICV_PUT_POINT();
431 
432                 tptr += step;
433                 a = (ep_corr * icvFilterTable[dist] >> 8) & 0xff;
434                 ICV_PUT_POINT();
435                 ICV_PUT_POINT();
436 
437                 tptr += step;
438                 a = (ep_corr * icvFilterTable[63 - dist] >> 8) & 0xff;
439                 ICV_PUT_POINT();
440                 ICV_PUT_POINT();
441 
442                 pt1.y += y_step;
443                 ptr += 3;
444                 scount++;
445                 ecount--;
446             }
447         }
448         else
449         {
450             ptr += (pt1.y >> XY_SHIFT) * step;
451 
452             while( ecount >= 0 )
453             {
454                 uchar *tptr = ptr + ((pt1.x >> XY_SHIFT) - 1) * 3;
455 
456                 int ep_corr = ep_table[(((scount >= 2) + 1) & (scount | 2)) * 3 +
457                                        (((ecount >= 2) + 1) & (ecount | 2))];
458                 int a, dist = (pt1.x >> (XY_SHIFT - 5)) & 31;
459 
460                 a = (ep_corr * icvFilterTable[dist + 32] >> 8) & 0xff;
461                 ICV_PUT_POINT();
462                 ICV_PUT_POINT();
463 
464                 tptr += 3;
465                 a = (ep_corr * icvFilterTable[dist] >> 8) & 0xff;
466                 ICV_PUT_POINT();
467                 ICV_PUT_POINT();
468 
469                 tptr += 3;
470                 a = (ep_corr * icvFilterTable[63 - dist] >> 8) & 0xff;
471                 ICV_PUT_POINT();
472                 ICV_PUT_POINT();
473 
474                 pt1.x += x_step;
475                 ptr += step;
476                 scount++;
477                 ecount--;
478             }
479         }
480         #undef ICV_PUT_POINT
481     }
482     else
483     {
484         #define  ICV_PUT_POINT()            \
485         {                                   \
486             _cb = tptr[0];                  \
487             _cb += ((cb - _cb)*a + 127)>> 8;\
488             tptr[0] = (uchar)_cb;           \
489         }
490 
491         if( ax > ay )
492         {
493             ptr += (pt1.x >> XY_SHIFT);
494 
495             while( ecount >= 0 )
496             {
497                 uchar *tptr = ptr + ((pt1.y >> XY_SHIFT) - 1) * step;
498 
499                 int ep_corr = ep_table[(((scount >= 2) + 1) & (scount | 2)) * 3 +
500                                        (((ecount >= 2) + 1) & (ecount | 2))];
501                 int a, dist = (pt1.y >> (XY_SHIFT - 5)) & 31;
502 
503                 a = (ep_corr * icvFilterTable[dist + 32] >> 8) & 0xff;
504                 ICV_PUT_POINT();
505                 ICV_PUT_POINT();
506 
507                 tptr += step;
508                 a = (ep_corr * icvFilterTable[dist] >> 8) & 0xff;
509                 ICV_PUT_POINT();
510                 ICV_PUT_POINT();
511 
512                 tptr += step;
513                 a = (ep_corr * icvFilterTable[63 - dist] >> 8) & 0xff;
514                 ICV_PUT_POINT();
515                 ICV_PUT_POINT();
516 
517                 pt1.y += y_step;
518                 ptr++;
519                 scount++;
520                 ecount--;
521             }
522         }
523         else
524         {
525             ptr += (pt1.y >> XY_SHIFT) * step;
526 
527             while( ecount >= 0 )
528             {
529                 uchar *tptr = ptr + ((pt1.x >> XY_SHIFT) - 1);
530 
531                 int ep_corr = ep_table[(((scount >= 2) + 1) & (scount | 2)) * 3 +
532                                        (((ecount >= 2) + 1) & (ecount | 2))];
533                 int a, dist = (pt1.x >> (XY_SHIFT - 5)) & 31;
534 
535                 a = (ep_corr * icvFilterTable[dist + 32] >> 8) & 0xff;
536                 ICV_PUT_POINT();
537                 ICV_PUT_POINT();
538 
539                 tptr++;
540                 a = (ep_corr * icvFilterTable[dist] >> 8) & 0xff;
541                 ICV_PUT_POINT();
542                 ICV_PUT_POINT();
543 
544                 tptr++;
545                 a = (ep_corr * icvFilterTable[63 - dist] >> 8) & 0xff;
546                 ICV_PUT_POINT();
547                 ICV_PUT_POINT();
548 
549                 pt1.x += x_step;
550                 ptr += step;
551                 scount++;
552                 ecount--;
553             }
554         }
555         #undef ICV_PUT_POINT
556     }
557 }
558 
559 
560 static void
icvLine2(CvMat * img,CvPoint pt1,CvPoint pt2,const void * color)561 icvLine2( CvMat* img, CvPoint pt1, CvPoint pt2, const void* color )
562 {
563     int dx, dy;
564     int ecount;
565     int ax, ay;
566     int i, j;
567     int x_step, y_step;
568     int cb = ((uchar*)color)[0];
569     int cg = ((uchar*)color)[1];
570     int cr = ((uchar*)color)[2];
571     int pix_size = CV_ELEM_SIZE( img->type );
572     uchar *ptr = (uchar*)(img->data.ptr), *tptr;
573     int step = img->step;
574     CvSize size = cvGetMatSize( img );
575 
576     //assert( img && (nch == 1 || nch == 3) && CV_MAT_DEPTH(img->type) == CV_8U );
577 
578     pt1.x -= XY_ONE*2;
579     pt1.y -= XY_ONE*2;
580     pt2.x -= XY_ONE*2;
581     pt2.y -= XY_ONE*2;
582     ptr += img->step*2 + 2*pix_size;
583 
584     size.width = ((size.width - 5) << XY_SHIFT) + 1;
585     size.height = ((size.height - 5) << XY_SHIFT) + 1;
586 
587     if( !cvClipLine( size, &pt1, &pt2 ))
588         return;
589 
590     dx = pt2.x - pt1.x;
591     dy = pt2.y - pt1.y;
592 
593     j = dx < 0 ? -1 : 0;
594     ax = (dx ^ j) - j;
595     i = dy < 0 ? -1 : 0;
596     ay = (dy ^ i) - i;
597 
598     if( ax > ay )
599     {
600         dx = ax;
601         dy = (dy ^ j) - j;
602         pt1.x ^= pt2.x & j;
603         pt2.x ^= pt1.x & j;
604         pt1.x ^= pt2.x & j;
605         pt1.y ^= pt2.y & j;
606         pt2.y ^= pt1.y & j;
607         pt1.y ^= pt2.y & j;
608 
609         x_step = XY_ONE;
610         y_step = (int) (((int64) dy << XY_SHIFT) / (ax | 1));
611         ecount = (pt2.x - pt1.x) >> XY_SHIFT;
612     }
613     else
614     {
615         dy = ay;
616         dx = (dx ^ i) - i;
617         pt1.x ^= pt2.x & i;
618         pt2.x ^= pt1.x & i;
619         pt1.x ^= pt2.x & i;
620         pt1.y ^= pt2.y & i;
621         pt2.y ^= pt1.y & i;
622         pt1.y ^= pt2.y & i;
623 
624         x_step = (int) (((int64) dx << XY_SHIFT) / (ay | 1));
625         y_step = XY_ONE;
626         ecount = (pt2.y - pt1.y) >> XY_SHIFT;
627     }
628 
629     pt1.x += (XY_ONE >> 1);
630     pt1.y += (XY_ONE >> 1);
631 
632     if( pix_size == 3 )
633     {
634         #define  ICV_PUT_POINT()    \
635         {                           \
636             tptr[0] = (uchar)cb;    \
637             tptr[1] = (uchar)cg;    \
638             tptr[2] = (uchar)cr;    \
639         }
640 
641         tptr = ptr + ((pt2.x + (XY_ONE >> 1))>> XY_SHIFT)*3 +
642             ((pt2.y + (XY_ONE >> 1)) >> XY_SHIFT)*step;
643         ICV_PUT_POINT();
644 
645         if( ax > ay )
646         {
647             ptr += (pt1.x >> XY_SHIFT) * 3;
648 
649             while( ecount >= 0 )
650             {
651                 tptr = ptr + (pt1.y >> XY_SHIFT) * step;
652                 ICV_PUT_POINT();
653                 pt1.y += y_step;
654                 ptr += 3;
655                 ecount--;
656             }
657         }
658         else
659         {
660             ptr += (pt1.y >> XY_SHIFT) * step;
661 
662             while( ecount >= 0 )
663             {
664                 tptr = ptr + (pt1.x >> XY_SHIFT) * 3;
665                 ICV_PUT_POINT();
666                 pt1.x += x_step;
667                 ptr += step;
668                 ecount--;
669             }
670         }
671 
672         #undef ICV_PUT_POINT
673     }
674     else if( pix_size == 1 )
675     {
676         #define  ICV_PUT_POINT()            \
677         {                                   \
678             tptr[0] = (uchar)cb;            \
679         }
680 
681         tptr = ptr + ((pt2.x + (XY_ONE >> 1))>> XY_SHIFT) +
682             ((pt2.y + (XY_ONE >> 1)) >> XY_SHIFT)*step;
683         ICV_PUT_POINT();
684 
685         if( ax > ay )
686         {
687             ptr += (pt1.x >> XY_SHIFT);
688 
689             while( ecount >= 0 )
690             {
691                 tptr = ptr + (pt1.y >> XY_SHIFT) * step;
692                 ICV_PUT_POINT();
693                 pt1.y += y_step;
694                 ptr++;
695                 ecount--;
696             }
697         }
698         else
699         {
700             ptr += (pt1.y >> XY_SHIFT) * step;
701 
702             while( ecount >= 0 )
703             {
704                 tptr = ptr + (pt1.x >> XY_SHIFT);
705                 ICV_PUT_POINT();
706                 pt1.x += x_step;
707                 ptr += step;
708                 ecount--;
709             }
710         }
711         #undef ICV_PUT_POINT
712     }
713     else
714     {
715         #define  ICV_PUT_POINT()                \
716             for( j = 0; j < pix_size; j++ )     \
717                 tptr[j] = ((uchar*)color)[j];
718 
719         tptr = ptr + ((pt2.x + (XY_ONE >> 1))>> XY_SHIFT)*pix_size +
720             ((pt2.y + (XY_ONE >> 1)) >> XY_SHIFT)*step;
721         ICV_PUT_POINT();
722 
723         if( ax > ay )
724         {
725             ptr += (pt1.x >> XY_SHIFT) * pix_size;
726 
727             while( ecount >= 0 )
728             {
729                 tptr = ptr + (pt1.y >> XY_SHIFT) * step;
730                 ICV_PUT_POINT();
731                 pt1.y += y_step;
732                 ptr += pix_size;
733                 ecount--;
734             }
735         }
736         else
737         {
738             ptr += (pt1.y >> XY_SHIFT) * step;
739 
740             while( ecount >= 0 )
741             {
742                 tptr = ptr + (pt1.x >> XY_SHIFT) * pix_size;
743                 ICV_PUT_POINT();
744                 pt1.x += x_step;
745                 ptr += step;
746                 ecount--;
747             }
748         }
749 
750         #undef ICV_PUT_POINT
751     }
752 }
753 
754 
755 /****************************************************************************************\
756 *                   Antialiazed Elliptic Arcs via Antialiazed Lines                      *
757 \****************************************************************************************/
758 
759 static const float icvSinTable[] =
760     { 0.0000000f, 0.0174524f, 0.0348995f, 0.0523360f, 0.0697565f, 0.0871557f,
761     0.1045285f, 0.1218693f, 0.1391731f, 0.1564345f, 0.1736482f, 0.1908090f,
762     0.2079117f, 0.2249511f, 0.2419219f, 0.2588190f, 0.2756374f, 0.2923717f,
763     0.3090170f, 0.3255682f, 0.3420201f, 0.3583679f, 0.3746066f, 0.3907311f,
764     0.4067366f, 0.4226183f, 0.4383711f, 0.4539905f, 0.4694716f, 0.4848096f,
765     0.5000000f, 0.5150381f, 0.5299193f, 0.5446390f, 0.5591929f, 0.5735764f,
766     0.5877853f, 0.6018150f, 0.6156615f, 0.6293204f, 0.6427876f, 0.6560590f,
767     0.6691306f, 0.6819984f, 0.6946584f, 0.7071068f, 0.7193398f, 0.7313537f,
768     0.7431448f, 0.7547096f, 0.7660444f, 0.7771460f, 0.7880108f, 0.7986355f,
769     0.8090170f, 0.8191520f, 0.8290376f, 0.8386706f, 0.8480481f, 0.8571673f,
770     0.8660254f, 0.8746197f, 0.8829476f, 0.8910065f, 0.8987940f, 0.9063078f,
771     0.9135455f, 0.9205049f, 0.9271839f, 0.9335804f, 0.9396926f, 0.9455186f,
772     0.9510565f, 0.9563048f, 0.9612617f, 0.9659258f, 0.9702957f, 0.9743701f,
773     0.9781476f, 0.9816272f, 0.9848078f, 0.9876883f, 0.9902681f, 0.9925462f,
774     0.9945219f, 0.9961947f, 0.9975641f, 0.9986295f, 0.9993908f, 0.9998477f,
775     1.0000000f, 0.9998477f, 0.9993908f, 0.9986295f, 0.9975641f, 0.9961947f,
776     0.9945219f, 0.9925462f, 0.9902681f, 0.9876883f, 0.9848078f, 0.9816272f,
777     0.9781476f, 0.9743701f, 0.9702957f, 0.9659258f, 0.9612617f, 0.9563048f,
778     0.9510565f, 0.9455186f, 0.9396926f, 0.9335804f, 0.9271839f, 0.9205049f,
779     0.9135455f, 0.9063078f, 0.8987940f, 0.8910065f, 0.8829476f, 0.8746197f,
780     0.8660254f, 0.8571673f, 0.8480481f, 0.8386706f, 0.8290376f, 0.8191520f,
781     0.8090170f, 0.7986355f, 0.7880108f, 0.7771460f, 0.7660444f, 0.7547096f,
782     0.7431448f, 0.7313537f, 0.7193398f, 0.7071068f, 0.6946584f, 0.6819984f,
783     0.6691306f, 0.6560590f, 0.6427876f, 0.6293204f, 0.6156615f, 0.6018150f,
784     0.5877853f, 0.5735764f, 0.5591929f, 0.5446390f, 0.5299193f, 0.5150381f,
785     0.5000000f, 0.4848096f, 0.4694716f, 0.4539905f, 0.4383711f, 0.4226183f,
786     0.4067366f, 0.3907311f, 0.3746066f, 0.3583679f, 0.3420201f, 0.3255682f,
787     0.3090170f, 0.2923717f, 0.2756374f, 0.2588190f, 0.2419219f, 0.2249511f,
788     0.2079117f, 0.1908090f, 0.1736482f, 0.1564345f, 0.1391731f, 0.1218693f,
789     0.1045285f, 0.0871557f, 0.0697565f, 0.0523360f, 0.0348995f, 0.0174524f,
790     0.0000000f, -0.0174524f, -0.0348995f, -0.0523360f, -0.0697565f, -0.0871557f,
791     -0.1045285f, -0.1218693f, -0.1391731f, -0.1564345f, -0.1736482f, -0.1908090f,
792     -0.2079117f, -0.2249511f, -0.2419219f, -0.2588190f, -0.2756374f, -0.2923717f,
793     -0.3090170f, -0.3255682f, -0.3420201f, -0.3583679f, -0.3746066f, -0.3907311f,
794     -0.4067366f, -0.4226183f, -0.4383711f, -0.4539905f, -0.4694716f, -0.4848096f,
795     -0.5000000f, -0.5150381f, -0.5299193f, -0.5446390f, -0.5591929f, -0.5735764f,
796     -0.5877853f, -0.6018150f, -0.6156615f, -0.6293204f, -0.6427876f, -0.6560590f,
797     -0.6691306f, -0.6819984f, -0.6946584f, -0.7071068f, -0.7193398f, -0.7313537f,
798     -0.7431448f, -0.7547096f, -0.7660444f, -0.7771460f, -0.7880108f, -0.7986355f,
799     -0.8090170f, -0.8191520f, -0.8290376f, -0.8386706f, -0.8480481f, -0.8571673f,
800     -0.8660254f, -0.8746197f, -0.8829476f, -0.8910065f, -0.8987940f, -0.9063078f,
801     -0.9135455f, -0.9205049f, -0.9271839f, -0.9335804f, -0.9396926f, -0.9455186f,
802     -0.9510565f, -0.9563048f, -0.9612617f, -0.9659258f, -0.9702957f, -0.9743701f,
803     -0.9781476f, -0.9816272f, -0.9848078f, -0.9876883f, -0.9902681f, -0.9925462f,
804     -0.9945219f, -0.9961947f, -0.9975641f, -0.9986295f, -0.9993908f, -0.9998477f,
805     -1.0000000f, -0.9998477f, -0.9993908f, -0.9986295f, -0.9975641f, -0.9961947f,
806     -0.9945219f, -0.9925462f, -0.9902681f, -0.9876883f, -0.9848078f, -0.9816272f,
807     -0.9781476f, -0.9743701f, -0.9702957f, -0.9659258f, -0.9612617f, -0.9563048f,
808     -0.9510565f, -0.9455186f, -0.9396926f, -0.9335804f, -0.9271839f, -0.9205049f,
809     -0.9135455f, -0.9063078f, -0.8987940f, -0.8910065f, -0.8829476f, -0.8746197f,
810     -0.8660254f, -0.8571673f, -0.8480481f, -0.8386706f, -0.8290376f, -0.8191520f,
811     -0.8090170f, -0.7986355f, -0.7880108f, -0.7771460f, -0.7660444f, -0.7547096f,
812     -0.7431448f, -0.7313537f, -0.7193398f, -0.7071068f, -0.6946584f, -0.6819984f,
813     -0.6691306f, -0.6560590f, -0.6427876f, -0.6293204f, -0.6156615f, -0.6018150f,
814     -0.5877853f, -0.5735764f, -0.5591929f, -0.5446390f, -0.5299193f, -0.5150381f,
815     -0.5000000f, -0.4848096f, -0.4694716f, -0.4539905f, -0.4383711f, -0.4226183f,
816     -0.4067366f, -0.3907311f, -0.3746066f, -0.3583679f, -0.3420201f, -0.3255682f,
817     -0.3090170f, -0.2923717f, -0.2756374f, -0.2588190f, -0.2419219f, -0.2249511f,
818     -0.2079117f, -0.1908090f, -0.1736482f, -0.1564345f, -0.1391731f, -0.1218693f,
819     -0.1045285f, -0.0871557f, -0.0697565f, -0.0523360f, -0.0348995f, -0.0174524f,
820     -0.0000000f, 0.0174524f, 0.0348995f, 0.0523360f, 0.0697565f, 0.0871557f,
821     0.1045285f, 0.1218693f, 0.1391731f, 0.1564345f, 0.1736482f, 0.1908090f,
822     0.2079117f, 0.2249511f, 0.2419219f, 0.2588190f, 0.2756374f, 0.2923717f,
823     0.3090170f, 0.3255682f, 0.3420201f, 0.3583679f, 0.3746066f, 0.3907311f,
824     0.4067366f, 0.4226183f, 0.4383711f, 0.4539905f, 0.4694716f, 0.4848096f,
825     0.5000000f, 0.5150381f, 0.5299193f, 0.5446390f, 0.5591929f, 0.5735764f,
826     0.5877853f, 0.6018150f, 0.6156615f, 0.6293204f, 0.6427876f, 0.6560590f,
827     0.6691306f, 0.6819984f, 0.6946584f, 0.7071068f, 0.7193398f, 0.7313537f,
828     0.7431448f, 0.7547096f, 0.7660444f, 0.7771460f, 0.7880108f, 0.7986355f,
829     0.8090170f, 0.8191520f, 0.8290376f, 0.8386706f, 0.8480481f, 0.8571673f,
830     0.8660254f, 0.8746197f, 0.8829476f, 0.8910065f, 0.8987940f, 0.9063078f,
831     0.9135455f, 0.9205049f, 0.9271839f, 0.9335804f, 0.9396926f, 0.9455186f,
832     0.9510565f, 0.9563048f, 0.9612617f, 0.9659258f, 0.9702957f, 0.9743701f,
833     0.9781476f, 0.9816272f, 0.9848078f, 0.9876883f, 0.9902681f, 0.9925462f,
834     0.9945219f, 0.9961947f, 0.9975641f, 0.9986295f, 0.9993908f, 0.9998477f,
835     1.0000000f
836 };
837 
838 
839 static void
icvSinCos(int angle,float * cosval,float * sinval)840 icvSinCos( int angle, float *cosval, float *sinval )
841 {
842     angle += (angle < 0 ? 360 : 0);
843     *sinval = icvSinTable[angle];
844     *cosval = icvSinTable[450 - angle];
845 }
846 
847 /*
848    constructs polygon that represents elliptic arc.
849 */
850 CV_IMPL int
cvEllipse2Poly(CvPoint center,CvSize axes,int angle,int arc_start,int arc_end,CvPoint * pts,int delta)851 cvEllipse2Poly( CvPoint center, CvSize axes, int angle,
852                 int arc_start, int arc_end, CvPoint* pts, int delta )
853 {
854     float alpha, beta;
855     double size_a = axes.width, size_b = axes.height;
856     double cx = center.x, cy = center.y;
857     CvPoint *pts_origin = pts;
858     int i;
859 
860     while( angle < 0 )
861         angle += 360;
862     while( angle > 360 )
863         angle -= 360;
864 
865     if( arc_start > arc_end )
866     {
867         i = arc_start;
868         arc_start = arc_end;
869         arc_end = i;
870     }
871     while( arc_start < 0 )
872     {
873         arc_start += 360;
874         arc_end += 360;
875     }
876     while( arc_end > 360 )
877     {
878         arc_end -= 360;
879         arc_start -= 360;
880     }
881     if( arc_end - arc_start > 360 )
882     {
883         arc_start = 0;
884         arc_end = 360;
885     }
886     icvSinCos( angle, &alpha, &beta );
887 
888     for( i = arc_start; i < arc_end + delta; i += delta )
889     {
890         double x, y;
891         angle = i;
892         if( angle > arc_end )
893             angle = arc_end;
894         if( angle < 0 )
895             angle += 360;
896 
897         x = size_a * icvSinTable[450-angle];
898         y = size_b * icvSinTable[angle];
899         pts->x = cvRound( cx + x * alpha - y * beta );
900         pts->y = cvRound( cy - x * beta - y * alpha );
901         pts += i == arc_start || pts->x != pts[-1].x || pts->y != pts[-1].y;
902     }
903 
904     i = (int)(pts - pts_origin);
905     for( ; i < 2; i++ )
906         pts_origin[i] = pts_origin[i-1];
907     return i;
908 }
909 
910 
911 static void
icvEllipseEx(CvMat * img,CvPoint center,CvSize axes,int angle,int arc_start,int arc_end,const void * color,int thickness,int line_type)912 icvEllipseEx( CvMat* img, CvPoint center, CvSize axes,
913               int angle, int arc_start, int arc_end,
914               const void* color, int thickness, int line_type )
915 {
916     CvMemStorage* st = 0;
917 
918     CV_FUNCNAME( "icvEllipseEx" );
919 
920     __BEGIN__;
921 
922     CvPoint v[1 << 8];
923     int count, delta;
924 
925     if( axes.width < 0 || axes.height < 0 )
926         CV_ERROR( CV_StsBadSize, "" );
927 
928     delta = (MAX(axes.width,axes.height)+(XY_ONE>>1))>>XY_SHIFT;
929     delta = delta < 3 ? 90 : delta < 10 ? 30 : delta < 15 ? 18 : 5;
930 
931     count = cvEllipse2Poly( center, axes, angle, arc_start, arc_end, v, delta );
932 
933     if( thickness >= 0 )
934     {
935         icvPolyLine( img, v, count, 0, color, thickness, line_type, XY_SHIFT );
936     }
937     else if( arc_end - arc_start >= 360 )
938     {
939         icvFillConvexPoly( img, v, count, color, line_type, XY_SHIFT );
940     }
941     else
942     {
943         CvContour* edges;
944         CvSeq vtx;
945         CvSeqBlock block;
946 
947         CV_CALL( st = cvCreateMemStorage( CV_DRAWING_STORAGE_BLOCK ));
948         CV_CALL( edges = (CvContour*)cvCreateSeq( 0, sizeof(CvContour), sizeof(CvPolyEdge), st ));
949         v[count++] = center;
950 
951         CV_CALL( cvMakeSeqHeaderForArray( CV_32SC2, sizeof(CvSeq), sizeof(CvPoint),
952                                           v, count, &vtx, &block ));
953 
954         CV_CALL( icvCollectPolyEdges( img, &vtx, edges, color, line_type, XY_SHIFT ));
955         CV_CALL( icvFillEdgeCollection( img, edges, color ));
956     }
957 
958     __END__;
959 
960     if( st )
961         cvReleaseMemStorage( &st );
962 }
963 
964 
965 /****************************************************************************************\
966 *                                Polygons filling                                        *
967 \****************************************************************************************/
968 
969 /* helper macros: filling horizontal row */
970 #define ICV_HLINE( ptr, xl, xr, color, pix_size )            \
971 {                                                            \
972     uchar* hline_ptr = (uchar*)(ptr) + (xl)*(pix_size);      \
973     uchar* hline_max_ptr = (uchar*)(ptr) + (xr)*(pix_size);  \
974                                                              \
975     for( ; hline_ptr <= hline_max_ptr; hline_ptr += (pix_size))\
976     {                                                        \
977         int hline_j;                                         \
978         for( hline_j = 0; hline_j < (pix_size); hline_j++ )  \
979         {                                                    \
980             hline_ptr[hline_j] = ((uchar*)color)[hline_j];   \
981         }                                                    \
982     }                                                        \
983 }
984 
985 
986 /* filling convex polygon. v - array of vertices, ntps - number of points */
987 static void
icvFillConvexPoly(CvMat * img,CvPoint * v,int npts,const void * color,int line_type,int shift)988 icvFillConvexPoly( CvMat* img, CvPoint *v, int npts, const void* color, int line_type, int shift )
989 {
990     struct
991     {
992         int idx, di;
993         int x, dx, ye;
994     }
995     edge[2];
996 
997     int delta = shift ? 1 << (shift - 1) : 0;
998     int i, y, imin = 0, left = 0, right = 1, x1, x2;
999     int edges = npts;
1000     int xmin, xmax, ymin, ymax;
1001     uchar* ptr = img->data.ptr;
1002     CvSize size = cvGetMatSize( img );
1003     int pix_size = CV_ELEM_SIZE(img->type);
1004     CvPoint p0;
1005     int delta1, delta2;
1006 
1007     if( line_type < CV_AA )
1008         delta1 = delta2 = XY_ONE >> 1;
1009         //delta1 = 0, delta2 = XY_ONE - 1;
1010     else
1011         delta1 = XY_ONE - 1, delta2 = 0;
1012 
1013     p0 = v[npts - 1];
1014     p0.x <<= XY_SHIFT - shift;
1015     p0.y <<= XY_SHIFT - shift;
1016 
1017     assert( 0 <= shift && shift <= XY_SHIFT );
1018     xmin = xmax = v[0].x;
1019     ymin = ymax = v[0].y;
1020 
1021     for( i = 0; i < npts; i++ )
1022     {
1023         CvPoint p = v[i];
1024         if( p.y < ymin )
1025         {
1026             ymin = p.y;
1027             imin = i;
1028         }
1029 
1030         ymax = MAX( ymax, p.y );
1031         xmax = MAX( xmax, p.x );
1032         xmin = MIN( xmin, p.x );
1033 
1034         p.x <<= XY_SHIFT - shift;
1035         p.y <<= XY_SHIFT - shift;
1036 
1037         if( line_type <= 8 )
1038         {
1039             if( shift == 0 )
1040             {
1041                 CvPoint pt0, pt1;
1042                 pt0.x = p0.x >> XY_SHIFT;
1043                 pt0.y = p0.y >> XY_SHIFT;
1044                 pt1.x = p.x >> XY_SHIFT;
1045                 pt1.y = p.y >> XY_SHIFT;
1046                 icvLine( img, pt0, pt1, color, line_type );
1047             }
1048             else
1049                 icvLine2( img, p0, p, color );
1050         }
1051         else
1052             icvLineAA( img, p0, p, color );
1053         p0 = p;
1054     }
1055 
1056     xmin = (xmin + delta) >> shift;
1057     xmax = (xmax + delta) >> shift;
1058     ymin = (ymin + delta) >> shift;
1059     ymax = (ymax + delta) >> shift;
1060 
1061     if( npts < 3 || xmax < 0 || ymax < 0 || xmin >= size.width || ymin >= size.height )
1062         return;
1063 
1064     ymax = MIN( ymax, size.height - 1 );
1065     edge[0].idx = edge[1].idx = imin;
1066 
1067     edge[0].ye = edge[1].ye = y = ymin;
1068     edge[0].di = 1;
1069     edge[1].di = npts - 1;
1070 
1071     ptr += img->step*y;
1072 
1073     do
1074     {
1075         if( line_type < CV_AA || y < ymax || y == ymin )
1076         {
1077             for( i = 0; i < 2; i++ )
1078             {
1079                 if( y >= edge[i].ye )
1080                 {
1081                     int idx = edge[i].idx, di = edge[i].di;
1082                     int xs = 0, xe, ye, ty = 0;
1083 
1084                     for(;;)
1085                     {
1086                         ty = (v[idx].y + delta) >> shift;
1087                         if( ty > y || edges == 0 )
1088                             break;
1089                         xs = v[idx].x;
1090                         idx += di;
1091                         idx -= ((idx < npts) - 1) & npts;   /* idx -= idx >= npts ? npts : 0 */
1092                         edges--;
1093                     }
1094 
1095                     ye = ty;
1096                     xs <<= XY_SHIFT - shift;
1097                     xe = v[idx].x << (XY_SHIFT - shift);
1098 
1099                     /* no more edges */
1100                     if( y >= ye )
1101                         return;
1102 
1103                     edge[i].ye = ye;
1104                     edge[i].dx = ((xe - xs)*2 + (ye - y)) / (2 * (ye - y));
1105                     edge[i].x = xs;
1106                     edge[i].idx = idx;
1107                 }
1108             }
1109         }
1110 
1111         if( edge[left].x > edge[right].x )
1112         {
1113             left ^= 1;
1114             right ^= 1;
1115         }
1116 
1117         x1 = edge[left].x;
1118         x2 = edge[right].x;
1119 
1120         if( y >= 0 )
1121         {
1122             int xx1 = (x1 + delta1) >> XY_SHIFT;
1123             int xx2 = (x2 + delta2) >> XY_SHIFT;
1124 
1125             if( xx2 >= 0 && xx1 < size.width )
1126             {
1127                 if( xx1 < 0 )
1128                     xx1 = 0;
1129                 if( xx2 >= size.width )
1130                     xx2 = size.width - 1;
1131                 ICV_HLINE( ptr, xx1, xx2, color, pix_size );
1132             }
1133         }
1134 
1135         x1 += edge[left].dx;
1136         x2 += edge[right].dx;
1137 
1138         edge[left].x = x1;
1139         edge[right].x = x2;
1140         ptr += img->step;
1141     }
1142     while( ++y <= ymax );
1143 }
1144 
1145 
1146 /******** Arbitrary polygon **********/
1147 
1148 static void
icvCollectPolyEdges(CvMat * img,CvSeq * v,CvContour * edges,const void * color,int line_type,int shift,CvPoint offset)1149 icvCollectPolyEdges( CvMat* img, CvSeq* v, CvContour* edges,
1150                      const void* color, int line_type, int shift,
1151                      CvPoint offset )
1152 {
1153     int  i, count = v->total;
1154     CvRect bounds = edges->rect;
1155     int delta = offset.y + (shift ? 1 << (shift - 1) : 0);
1156     int elem_type = CV_MAT_TYPE(v->flags);
1157 
1158     CvSeqReader reader;
1159     CvSeqWriter writer;
1160 
1161     cvStartReadSeq( v, &reader );
1162     cvStartAppendToSeq( (CvSeq*)edges, &writer );
1163 
1164     for( i = 0; i < count; i++ )
1165     {
1166         CvPoint pt0, pt1, t0, t1;
1167         CvPolyEdge edge;
1168         CV_READ_EDGE( pt0, pt1, reader );
1169 
1170         if( elem_type == CV_32SC2 )
1171         {
1172             pt0.x = (pt0.x + offset.x) << (XY_SHIFT - shift);
1173             pt0.y = (pt0.y + delta) >> shift;
1174             pt1.x = (pt1.x + offset.x) << (XY_SHIFT - shift);
1175             pt1.y = (pt1.y + delta) >> shift;
1176         }
1177         else
1178         {
1179             Cv32suf x, y;
1180             assert( shift == 0 );
1181 
1182             x.i = pt0.x; y.i = pt0.y;
1183             pt0.x = cvRound((x.f + offset.x) * XY_ONE);
1184             pt0.y = cvRound(y.f + offset.y);
1185             x.i = pt1.x; y.i = pt1.y;
1186             pt1.x = cvRound((x.f + offset.x) * XY_ONE);
1187             pt1.y = cvRound(y.f + offset.y);
1188         }
1189 
1190         if( line_type < CV_AA )
1191         {
1192             t0.y = pt0.y; t1.y = pt1.y;
1193             t0.x = (pt0.x + (XY_ONE >> 1)) >> XY_SHIFT;
1194             t1.x = (pt1.x + (XY_ONE >> 1)) >> XY_SHIFT;
1195             icvLine( img, t0, t1, color, line_type );
1196         }
1197         else
1198         {
1199             t0.x = pt0.x; t1.x = pt1.x;
1200             t0.y = pt0.y << XY_SHIFT;
1201             t1.y = pt1.y << XY_SHIFT;
1202             icvLineAA( img, t0, t1, color );
1203         }
1204 
1205         if( pt0.y == pt1.y )
1206             continue;
1207 
1208         if( pt0.y > pt1.y )
1209             CV_SWAP( pt0, pt1, t0 );
1210 
1211         bounds.y = MIN( bounds.y, pt0.y );
1212         bounds.height = MAX( bounds.height, pt1.y );
1213 
1214         if( pt0.x < pt1.x )
1215         {
1216             bounds.x = MIN( bounds.x, pt0.x );
1217             bounds.width = MAX( bounds.width, pt1.x );
1218         }
1219         else
1220         {
1221             bounds.x = MIN( bounds.x, pt1.x );
1222             bounds.width = MAX( bounds.width, pt0.x );
1223         }
1224 
1225         edge.y0 = pt0.y;
1226         edge.y1 = pt1.y;
1227         edge.x = pt0.x;
1228         edge.dx = (pt1.x - pt0.x) / (pt1.y - pt0.y);
1229         assert( edge.y0 < edge.y1 );
1230 
1231         CV_WRITE_SEQ_ELEM( edge, writer );
1232     }
1233 
1234     edges->rect = bounds;
1235     cvEndWriteSeq( &writer );
1236 }
1237 
1238 static int
icvCmpEdges(const void * _e1,const void * _e2,void *)1239 icvCmpEdges( const void* _e1, const void* _e2, void* /*userdata*/ )
1240 {
1241     CvPolyEdge *e1 = (CvPolyEdge*)_e1, *e2 = (CvPolyEdge*)_e2;
1242     return e1->y0 - e2->y0 ? e1->y0 - e2->y0 :
1243            e1->x - e2->x ? e1->x - e2->x : e1->dx - e2->dx;
1244 }
1245 
1246 /**************** helper macros and functions for sequence/contour processing ***********/
1247 
1248 static void
icvFillEdgeCollection(CvMat * img,CvContour * edges,const void * color)1249 icvFillEdgeCollection( CvMat* img, CvContour* edges, const void* color )
1250 {
1251     CvPolyEdge tmp;
1252     int i, y, total = edges->total;
1253     CvSeqReader reader;
1254     CvSize size = cvGetMatSize(img);
1255     CvPolyEdge* e;
1256     int y_max = INT_MIN;
1257     int pix_size = CV_ELEM_SIZE(img->type);
1258 
1259     __BEGIN__;
1260 
1261     memset( &tmp, 0, sizeof(tmp));
1262 
1263     /* check parameters */
1264     if( edges->total < 2 || edges->rect.height < 0 || edges->rect.y >= size.height ||
1265         edges->rect.width < 0 || edges->rect.x >= size.width )
1266         EXIT;
1267 
1268     cvSeqSort( (CvSeq*)edges, icvCmpEdges, 0 );
1269     cvStartReadSeq( (CvSeq*)edges, &reader );
1270 
1271 #ifdef _DEBUG
1272     e = &tmp;
1273     tmp.y0 = INT_MIN;
1274 #endif
1275 
1276     for( i = 0; i < total; i++ )
1277     {
1278         CvPolyEdge* e1 = (CvPolyEdge*)(reader.ptr);
1279 
1280 #ifdef _DEBUG
1281         assert( e1->y0 < e1->y1 && (i == 0 || icvCmpEdges( e, e1, 0 ) <= 0) );
1282         e = e1;
1283 #endif
1284         y_max = MAX( y_max, e1->y1 );
1285 
1286         CV_NEXT_SEQ_ELEM( sizeof(CvPolyEdge), reader );
1287     }
1288 
1289     /* start drawing */
1290     tmp.y0 = INT_MAX;
1291     cvSeqPush( (CvSeq*)edges, &tmp );
1292 
1293     i = 0;
1294     tmp.next = 0;
1295     cvStartReadSeq( (CvSeq*)edges, &reader );
1296     e = (CvPolyEdge*)(reader.ptr);
1297     y_max = MIN( y_max, size.height );
1298 
1299     for( y = e->y0; y < y_max; y++ )
1300     {
1301         CvPolyEdge *last, *prelast, *keep_prelast;
1302         int sort_flag = 0;
1303         int draw = 0;
1304         int clipline = y < 0;
1305 
1306         prelast = &tmp;
1307         last = tmp.next;
1308         while( last || e->y0 == y )
1309         {
1310             if( last && last->y1 == y )
1311             {
1312                 /* exlude edge if y reachs its lower point */
1313                 prelast->next = last->next;
1314                 last = last->next;
1315                 continue;
1316             }
1317             keep_prelast = prelast;
1318             if( last && (e->y0 > y || last->x < e->x) )
1319             {
1320                 /* go to the next edge in active list */
1321                 prelast = last;
1322                 last = last->next;
1323             }
1324             else if( i < total )
1325             {
1326                 /* insert new edge into active list if y reachs its upper point */
1327                 prelast->next = e;
1328                 e->next = last;
1329                 prelast = e;
1330                 CV_NEXT_SEQ_ELEM( edges->elem_size, reader );
1331                 e = (CvPolyEdge*)(reader.ptr);
1332                 i++;
1333             }
1334             else
1335                 break;
1336 
1337             if( draw )
1338             {
1339                 if( !clipline )
1340                 {
1341                     /* convert x's from fixed-point to image coordinates */
1342                     uchar *timg = (uchar*)(img->data.ptr) + y * img->step;
1343                     int x1 = keep_prelast->x;
1344                     int x2 = prelast->x;
1345 
1346                     if( x1 > x2 )
1347                     {
1348                         int t = x1;
1349 
1350                         x1 = x2;
1351                         x2 = t;
1352                     }
1353 
1354                     x1 = (x1 + XY_ONE - 1) >> XY_SHIFT;
1355                     x2 = x2 >> XY_SHIFT;
1356 
1357                     /* clip and draw the line */
1358                     if( x1 < size.width && x2 >= 0 )
1359                     {
1360                         if( x1 < 0 )
1361                             x1 = 0;
1362                         if( x2 >= size.width )
1363                             x2 = size.width - 1;
1364                         ICV_HLINE( timg, x1, x2, color, pix_size );
1365                     }
1366                 }
1367                 keep_prelast->x += keep_prelast->dx;
1368                 prelast->x += prelast->dx;
1369             }
1370             draw ^= 1;
1371         }
1372 
1373         /* sort edges (bubble sort on list) */
1374         keep_prelast = 0;
1375 
1376         do
1377         {
1378             prelast = &tmp;
1379             last = tmp.next;
1380 
1381             while( last != keep_prelast && last->next != 0 )
1382             {
1383                 CvPolyEdge *te = last->next;
1384 
1385                 /* swap edges */
1386                 if( last->x > te->x )
1387                 {
1388                     prelast->next = te;
1389                     last->next = te->next;
1390                     te->next = last;
1391                     prelast = te;
1392                     sort_flag = 1;
1393                 }
1394                 else
1395                 {
1396                     prelast = last;
1397                     last = te;
1398                 }
1399             }
1400             keep_prelast = prelast;
1401         }
1402         while( sort_flag && keep_prelast != tmp.next && keep_prelast != &tmp );
1403     }
1404 
1405     __END__;
1406 }
1407 
1408 
1409 /* draws simple or filled circle */
1410 static void
icvCircle(CvMat * img,CvPoint center,int radius,const void * color,int fill)1411 icvCircle( CvMat* img, CvPoint center, int radius, const void* color, int fill )
1412 {
1413     CvSize size = cvGetMatSize( img );
1414     int step = img->step;
1415     int pix_size = CV_ELEM_SIZE(img->type);
1416     uchar* ptr = (uchar*)(img->data.ptr);
1417     int err = 0, dx = radius, dy = 0, plus = 1, minus = (radius << 1) - 1;
1418     int inside = center.x >= radius && center.x < size.width - radius &&
1419         center.y >= radius && center.y < size.height - radius;
1420 
1421     #define ICV_PUT_POINT( ptr, x )     \
1422         CV_MEMCPY_CHAR( ptr + (x)*pix_size, color, pix_size );
1423 
1424     while( dx >= dy )
1425     {
1426         int mask;
1427         int y11 = center.y - dy, y12 = center.y + dy, y21 = center.y - dx, y22 = center.y + dx;
1428         int x11 = center.x - dx, x12 = center.x + dx, x21 = center.x - dy, x22 = center.x + dy;
1429 
1430         if( inside )
1431         {
1432             uchar *tptr0 = ptr + y11 * step;
1433             uchar *tptr1 = ptr + y12 * step;
1434 
1435             if( !fill )
1436             {
1437                 ICV_PUT_POINT( tptr0, x11 );
1438                 ICV_PUT_POINT( tptr1, x11 );
1439                 ICV_PUT_POINT( tptr0, x12 );
1440                 ICV_PUT_POINT( tptr1, x12 );
1441             }
1442             else
1443             {
1444                 ICV_HLINE( tptr0, x11, x12, color, pix_size );
1445                 ICV_HLINE( tptr1, x11, x12, color, pix_size );
1446             }
1447 
1448             tptr0 = ptr + y21 * step;
1449             tptr1 = ptr + y22 * step;
1450 
1451             if( !fill )
1452             {
1453                 ICV_PUT_POINT( tptr0, x21 );
1454                 ICV_PUT_POINT( tptr1, x21 );
1455                 ICV_PUT_POINT( tptr0, x22 );
1456                 ICV_PUT_POINT( tptr1, x22 );
1457             }
1458             else
1459             {
1460                 ICV_HLINE( tptr0, x21, x22, color, pix_size );
1461                 ICV_HLINE( tptr1, x21, x22, color, pix_size );
1462             }
1463         }
1464         else if( x11 < size.width && x12 >= 0 && y21 < size.height && y22 >= 0 )
1465         {
1466             if( fill )
1467             {
1468                 x11 = MAX( x11, 0 );
1469                 x12 = MIN( x12, size.width - 1 );
1470             }
1471 
1472             if( (unsigned)y11 < (unsigned)size.height )
1473             {
1474                 uchar *tptr = ptr + y11 * step;
1475 
1476                 if( !fill )
1477                 {
1478                     if( x11 >= 0 )
1479                         ICV_PUT_POINT( tptr, x11 );
1480                     if( x12 < size.width )
1481                         ICV_PUT_POINT( tptr, x12 );
1482                 }
1483                 else
1484                     ICV_HLINE( tptr, x11, x12, color, pix_size );
1485             }
1486 
1487             if( (unsigned)y12 < (unsigned)size.height )
1488             {
1489                 uchar *tptr = ptr + y12 * step;
1490 
1491                 if( !fill )
1492                 {
1493                     if( x11 >= 0 )
1494                         ICV_PUT_POINT( tptr, x11 );
1495                     if( x12 < size.width )
1496                         ICV_PUT_POINT( tptr, x12 );
1497                 }
1498                 else
1499                     ICV_HLINE( tptr, x11, x12, color, pix_size );
1500             }
1501 
1502             if( x21 < size.width && x22 >= 0 )
1503             {
1504                 if( fill )
1505                 {
1506                     x21 = MAX( x21, 0 );
1507                     x22 = MIN( x22, size.width - 1 );
1508                 }
1509 
1510                 if( (unsigned)y21 < (unsigned)size.height )
1511                 {
1512                     uchar *tptr = ptr + y21 * step;
1513 
1514                     if( !fill )
1515                     {
1516                         if( x21 >= 0 )
1517                             ICV_PUT_POINT( tptr, x21 );
1518                         if( x22 < size.width )
1519                             ICV_PUT_POINT( tptr, x22 );
1520                     }
1521                     else
1522                         ICV_HLINE( tptr, x21, x22, color, pix_size );
1523                 }
1524 
1525                 if( (unsigned)y22 < (unsigned)size.height )
1526                 {
1527                     uchar *tptr = ptr + y22 * step;
1528 
1529                     if( !fill )
1530                     {
1531                         if( x21 >= 0 )
1532                             ICV_PUT_POINT( tptr, x21 );
1533                         if( x22 < size.width )
1534                             ICV_PUT_POINT( tptr, x22 );
1535                     }
1536                     else
1537                         ICV_HLINE( tptr, x21, x22, color, pix_size );
1538                 }
1539             }
1540         }
1541         dy++;
1542         err += plus;
1543         plus += 2;
1544 
1545         mask = (err <= 0) - 1;
1546 
1547         err -= minus & mask;
1548         dx += mask;
1549         minus -= mask & 2;
1550     }
1551 
1552     #undef  ICV_PUT_POINT
1553 }
1554 
1555 
1556 static void
icvThickLine(CvMat * img,CvPoint p0,CvPoint p1,const void * color,int thickness,int line_type,int flags,int shift)1557 icvThickLine( CvMat* img, CvPoint p0, CvPoint p1, const void* color,
1558               int thickness, int line_type, int flags, int shift )
1559 {
1560     static const double INV_XY_ONE = 1./XY_ONE;
1561 
1562     p0.x <<= XY_SHIFT - shift;
1563     p0.y <<= XY_SHIFT - shift;
1564     p1.x <<= XY_SHIFT - shift;
1565     p1.y <<= XY_SHIFT - shift;
1566 
1567     if( thickness <= 1 )
1568     {
1569         if( line_type < CV_AA )
1570         {
1571             if( line_type == 1 || line_type == 4 || shift == 0 )
1572             {
1573                 p0.x = (p0.x + (XY_ONE>>1)) >> XY_SHIFT;
1574                 p0.y = (p0.y + (XY_ONE>>1)) >> XY_SHIFT;
1575                 p1.x = (p1.x + (XY_ONE>>1)) >> XY_SHIFT;
1576                 p1.y = (p1.y + (XY_ONE>>1)) >> XY_SHIFT;
1577                 icvLine( img, p0, p1, color, line_type );
1578             }
1579             else
1580                 icvLine2( img, p0, p1, color );
1581         }
1582         else
1583             icvLineAA( img, p0, p1, color );
1584     }
1585     else
1586     {
1587         CvPoint pt[4], dp = {0,0};
1588         double dx = (p0.x - p1.x)*INV_XY_ONE, dy = (p1.y - p0.y)*INV_XY_ONE;
1589         double r = dx * dx + dy * dy;
1590         int i;
1591         thickness <<= XY_SHIFT - 1;
1592 
1593         if( fabs(r) > DBL_EPSILON )
1594         {
1595             r = thickness * cvInvSqrt( (float) r );
1596             dp.x = cvRound( dy * r );
1597             dp.y = cvRound( dx * r );
1598         }
1599 
1600         pt[0].x = p0.x + dp.x;
1601         pt[0].y = p0.y + dp.y;
1602         pt[1].x = p0.x - dp.x;
1603         pt[1].y = p0.y - dp.y;
1604         pt[2].x = p1.x - dp.x;
1605         pt[2].y = p1.y - dp.y;
1606         pt[3].x = p1.x + dp.x;
1607         pt[3].y = p1.y + dp.y;
1608 
1609         icvFillConvexPoly( img, pt, 4, color, line_type, XY_SHIFT );
1610 
1611         for( i = 0; i < 2; i++ )
1612         {
1613             if( flags & (i+1) )
1614             {
1615                 if( line_type < CV_AA )
1616                 {
1617                     CvPoint center;
1618                     center.x = (p0.x + (XY_ONE>>1)) >> XY_SHIFT;
1619                     center.y = (p0.y + (XY_ONE>>1)) >> XY_SHIFT;
1620                     icvCircle( img, center, thickness >> XY_SHIFT, color, 1 );
1621                 }
1622                 else
1623                 {
1624                     icvEllipseEx( img, p0, cvSize(thickness, thickness),
1625                                   0, 0, 360, color, -1, line_type );
1626                 }
1627             }
1628             p0 = p1;
1629         }
1630     }
1631 }
1632 
1633 
1634 static void
icvPolyLine(CvMat * img,CvPoint * v,int count,int is_closed,const void * color,int thickness,int line_type,int shift)1635 icvPolyLine( CvMat* img, CvPoint *v, int count, int is_closed,
1636              const void* color, int thickness,
1637              int line_type, int shift )
1638 {
1639     CV_FUNCNAME("icvPolyLine");
1640 
1641     __BEGIN__;
1642 
1643     if( count > 0 )
1644     {
1645         int i = is_closed ? count - 1 : 0;
1646         int flags = 2 + !is_closed;
1647         CvPoint p0;
1648         assert( 0 <= shift && shift <= XY_SHIFT );
1649         assert( img && thickness >= 0 );
1650         assert( v && count >= 0 );
1651 
1652         if( !v )
1653             CV_ERROR( CV_StsNullPtr, "" );
1654 
1655         p0 = v[i];
1656         for( i = !is_closed; i < count; i++ )
1657         {
1658             CvPoint p = v[i];
1659             icvThickLine( img, p0, p, color, thickness, line_type, flags, shift );
1660             p0 = p;
1661             flags = 2;
1662         }
1663     }
1664 
1665     __END__;
1666 }
1667 
1668 /****************************************************************************************\
1669 *                              External functions                                        *
1670 \****************************************************************************************/
1671 
cvColorToScalar(double packed_color,int type)1672 CV_IMPL CvScalar cvColorToScalar( double packed_color, int type )
1673 {
1674     CvScalar scalar;
1675 
1676     if( CV_MAT_DEPTH( type ) == CV_8U )
1677     {
1678         int icolor = cvRound( packed_color );
1679         if( CV_MAT_CN( type ) > 1 )
1680         {
1681             scalar.val[0] = icolor & 255;
1682             scalar.val[1] = (icolor >> 8) & 255;
1683             scalar.val[2] = (icolor >> 16) & 255;
1684             scalar.val[3] = (icolor >> 24) & 255;
1685         }
1686         else
1687         {
1688             scalar.val[0] = CV_CAST_8U( icolor );
1689             scalar.val[1] = scalar.val[2] = scalar.val[3] = 0;
1690         }
1691     }
1692     else if( CV_MAT_DEPTH( type ) == CV_8S )
1693     {
1694         int icolor = cvRound( packed_color );
1695         if( CV_MAT_CN( type ) > 1 )
1696         {
1697             scalar.val[0] = (char)icolor;
1698             scalar.val[1] = (char)(icolor >> 8);
1699             scalar.val[2] = (char)(icolor >> 16);
1700             scalar.val[3] = (char)(icolor >> 24);
1701         }
1702         else
1703         {
1704             scalar.val[0] = CV_CAST_8S( icolor );
1705             scalar.val[1] = scalar.val[2] = scalar.val[3] = 0;
1706         }
1707     }
1708     else
1709     {
1710         int cn = CV_MAT_CN( type );
1711         switch( cn )
1712         {
1713         case 1:
1714             scalar.val[0] = packed_color;
1715             scalar.val[1] = scalar.val[2] = scalar.val[3] = 0;
1716             break;
1717         case 2:
1718             scalar.val[0] = scalar.val[1] = packed_color;
1719             scalar.val[2] = scalar.val[3] = 0;
1720             break;
1721         case 3:
1722             scalar.val[0] = scalar.val[1] = scalar.val[2] = packed_color;
1723             scalar.val[3] = 0;
1724             break;
1725         default:
1726             scalar.val[0] = scalar.val[1] =
1727                 scalar.val[2] = scalar.val[3] = packed_color;
1728             break;
1729         }
1730     }
1731 
1732     return scalar;
1733 }
1734 
1735 
1736 CV_IMPL void
cvLine(void * img,CvPoint pt1,CvPoint pt2,CvScalar color,int thickness,int line_type,int shift)1737 cvLine( void* img, CvPoint pt1, CvPoint pt2, CvScalar color,
1738         int thickness, int line_type, int shift )
1739 {
1740     CV_FUNCNAME( "cvLine" );
1741 
1742     __BEGIN__;
1743 
1744     int coi = 0;
1745     CvMat stub, *mat = (CvMat*)img;
1746     double buf[4];
1747 
1748     CV_CALL( mat = cvGetMat( img, &stub, &coi ));
1749 
1750     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
1751         line_type = 8;
1752 
1753     if( coi != 0 )
1754         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
1755 
1756     if( (unsigned)thickness > 255  )
1757         CV_ERROR( CV_StsOutOfRange, "" );
1758 
1759     if( shift < 0 || XY_SHIFT < shift )
1760         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
1761 
1762     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
1763     icvThickLine( mat, pt1, pt2, buf, thickness, line_type, 3, shift );
1764 
1765     __END__;
1766 }
1767 
1768 
1769 CV_IMPL void
cvRectangle(void * img,CvPoint pt1,CvPoint pt2,CvScalar color,int thickness,int line_type,int shift)1770 cvRectangle( void* img, CvPoint pt1, CvPoint pt2,
1771              CvScalar color, int thickness,
1772              int line_type, int shift )
1773 {
1774     CvPoint pt[4];
1775 
1776     CV_FUNCNAME("cvRectangle");
1777 
1778     __BEGIN__;
1779 
1780     int coi = 0;
1781     CvMat stub, *mat = (CvMat*)img;
1782     double buf[4];
1783 
1784     if( thickness > 255 )
1785         CV_ERROR( CV_StsOutOfRange, "" );
1786 
1787     CV_CALL( mat = cvGetMat( img, &stub, &coi ));
1788 
1789     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
1790         line_type = 8;
1791 
1792     if( coi != 0 )
1793         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
1794 
1795     if( shift < 0 || XY_SHIFT < shift )
1796         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
1797 
1798     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
1799 
1800     pt[0] = pt1;
1801     pt[1].x = pt2.x;
1802     pt[1].y = pt1.y;
1803     pt[2] = pt2;
1804     pt[3].x = pt1.x;
1805     pt[3].y = pt2.y;
1806 
1807     if( thickness >= 0 )
1808         icvPolyLine( mat, pt, 4, 1, buf, thickness, line_type, shift );
1809     else
1810         icvFillConvexPoly( mat, pt, 4, buf, line_type, shift );
1811 
1812     __END__;
1813 }
1814 
1815 
1816 CV_IMPL void
cvCircle(void * img,CvPoint center,int radius,CvScalar color,int thickness,int line_type,int shift)1817 cvCircle( void *img, CvPoint center, int radius,
1818           CvScalar color, int thickness, int line_type, int shift )
1819 {
1820     CV_FUNCNAME( "cvCircle" );
1821 
1822     __BEGIN__;
1823 
1824     int coi = 0;
1825     CvMat stub, *mat = (CvMat*)img;
1826     double buf[4];
1827 
1828     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
1829 
1830     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
1831         line_type = 8;
1832 
1833     if( coi != 0 )
1834         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
1835 
1836     if( radius < 0 )
1837         CV_ERROR( CV_StsOutOfRange, "" );
1838 
1839     if( thickness > 255 )
1840         CV_ERROR( CV_StsOutOfRange, "" );
1841 
1842     if( shift < 0 || XY_SHIFT < shift )
1843         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
1844 
1845     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
1846 
1847     if( thickness > 1 || line_type >= CV_AA )
1848     {
1849         center.x <<= XY_SHIFT - shift;
1850         center.y <<= XY_SHIFT - shift;
1851         radius <<= XY_SHIFT - shift;
1852         icvEllipseEx( mat, center, cvSize( radius, radius ),
1853                       0, 0, 360, buf, thickness, line_type );
1854     }
1855     else
1856     {
1857         icvCircle( mat, center, radius, buf, thickness < 0 );
1858     }
1859 
1860     __END__;
1861 }
1862 
1863 
1864 CV_IMPL void
cvEllipse(void * img,CvPoint center,CvSize axes,double angle,double start_angle,double end_angle,CvScalar color,int thickness,int line_type,int shift)1865 cvEllipse( void *img, CvPoint center, CvSize axes,
1866            double angle, double start_angle, double end_angle,
1867            CvScalar color, int thickness, int line_type, int shift )
1868 {
1869     CV_FUNCNAME( "cvEllipse" );
1870 
1871     __BEGIN__;
1872 
1873     int coi = 0;
1874     CvMat stub, *mat = (CvMat*)img;
1875     double buf[4];
1876 
1877     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
1878 
1879     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
1880         line_type = 8;
1881 
1882     if( coi != 0 )
1883         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
1884 
1885     if( axes.width < 0 || axes.height < 0 )
1886         CV_ERROR( CV_StsOutOfRange, "" );
1887 
1888     if( thickness > 255 )
1889         CV_ERROR( CV_StsOutOfRange, "" );
1890 
1891     if( shift < 0 || XY_SHIFT < shift )
1892         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
1893 
1894     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
1895 
1896     {
1897         int _angle = cvRound(angle);
1898         int _start_angle = cvRound(start_angle);
1899         int _end_angle = cvRound(end_angle);
1900         center.x <<= XY_SHIFT - shift;
1901         center.y <<= XY_SHIFT - shift;
1902         axes.width <<= XY_SHIFT - shift;
1903         axes.height <<= XY_SHIFT - shift;
1904 
1905         CV_CALL( icvEllipseEx( mat, center, axes, _angle, _start_angle,
1906                                _end_angle, buf, thickness, line_type ));
1907     }
1908 
1909     __END__;
1910 }
1911 
1912 
1913 CV_IMPL void
cvFillConvexPoly(void * img,CvPoint * pts,int npts,CvScalar color,int line_type,int shift)1914 cvFillConvexPoly( void *img, CvPoint *pts, int npts, CvScalar color, int line_type, int shift )
1915 {
1916     CV_FUNCNAME( "cvFillConvexPoly" );
1917 
1918     __BEGIN__;
1919 
1920     int coi = 0;
1921     CvMat stub, *mat = (CvMat*)img;
1922     double buf[4];
1923 
1924     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
1925 
1926     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
1927         line_type = 8;
1928 
1929     if( coi != 0 )
1930         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
1931 
1932     if( !pts )
1933         CV_ERROR( CV_StsNullPtr, "" );
1934 
1935     if( npts <= 0 )
1936         CV_ERROR( CV_StsOutOfRange, "" );
1937 
1938     if( shift < 0 || XY_SHIFT < shift )
1939         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
1940 
1941     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
1942     icvFillConvexPoly( mat, pts, npts, buf, line_type, shift );
1943 
1944     __END__;
1945 }
1946 
1947 
1948 CV_IMPL void
cvFillPoly(void * img,CvPoint ** pts,int * npts,int contours,CvScalar color,int line_type,int shift)1949 cvFillPoly( void *img, CvPoint **pts, int *npts, int contours,
1950             CvScalar color, int line_type, int shift )
1951 {
1952     CvMemStorage* st = 0;
1953 
1954     CV_FUNCNAME( "cvFillPoly" );
1955 
1956     __BEGIN__;
1957 
1958     int coi = 0;
1959     CvMat stub, *mat = (CvMat*)img;
1960     double buf[4];
1961 
1962     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
1963 
1964     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
1965         line_type = 8;
1966 
1967     if( coi != 0 )
1968         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
1969 
1970     if( contours <= 0 )
1971         CV_ERROR( CV_StsBadArg, "" );
1972 
1973     if( !pts )
1974         CV_ERROR( CV_StsNullPtr, "" );
1975 
1976     if( !npts )
1977         CV_ERROR( CV_StsNullPtr, "" );
1978 
1979     if( shift < 0 || XY_SHIFT < shift )
1980         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
1981 
1982     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
1983 
1984     {
1985         CvContour* edges = 0;
1986         CvSeq vtx;
1987         CvSeqBlock block;
1988 
1989         CV_CALL( st = cvCreateMemStorage( CV_DRAWING_STORAGE_BLOCK ));
1990         CV_CALL( edges = (CvContour*)cvCreateSeq( 0, sizeof(CvContour),
1991                                                   sizeof(CvPolyEdge), st ));
1992 
1993         for( int i = 0; i < contours; i++ )
1994         {
1995             if( !pts[i] )
1996                 CV_ERROR( CV_StsNullPtr, "" );
1997 
1998             if( npts[i] < 0 )
1999                 CV_ERROR( CV_StsOutOfRange, "" );
2000 
2001             cvMakeSeqHeaderForArray( CV_32SC2, sizeof(CvSeq), sizeof(CvPoint),
2002                                      pts[i], npts[i], &vtx, &block );
2003 
2004             CV_CALL( icvCollectPolyEdges( mat, &vtx, edges, buf, line_type, shift ));
2005         }
2006 
2007         CV_CALL( icvFillEdgeCollection( mat, edges, buf ));
2008     }
2009 
2010     __END__;
2011 
2012     cvReleaseMemStorage( &st );
2013 }
2014 
2015 
2016 
2017 CV_IMPL void
cvPolyLine(void * img,CvPoint ** pts,int * npts,int contours,int closed,CvScalar color,int thickness,int line_type,int shift)2018 cvPolyLine( void *img, CvPoint **pts, int *npts,
2019             int contours, int closed, CvScalar color,
2020             int thickness, int line_type, int shift )
2021 {
2022     CV_FUNCNAME( "cvPolyLine" );
2023 
2024     __BEGIN__;
2025 
2026     int coi = 0, i;
2027     CvMat stub, *mat = (CvMat*)img;
2028     double buf[4];
2029 
2030     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
2031 
2032     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
2033         line_type = 8;
2034 
2035     if( coi != 0 )
2036         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
2037 
2038     if( contours <= 0 )
2039         CV_ERROR( CV_StsBadArg, "" );
2040 
2041     if( thickness < -1 || thickness > 255 )
2042         CV_ERROR( CV_StsBadArg, "" );
2043 
2044     if( !pts )
2045         CV_ERROR( CV_StsNullPtr, "" );
2046 
2047     if( !npts )
2048         CV_ERROR( CV_StsNullPtr, "" );
2049 
2050     if( shift < 0 || XY_SHIFT < shift )
2051         CV_ERROR( CV_StsOutOfRange, "shift must be between 0 and 16" );
2052 
2053     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
2054 
2055     for( i = 0; i < contours; i++ )
2056         icvPolyLine( mat, pts[i], npts[i], closed, buf, thickness, line_type, shift );
2057 
2058     __END__;
2059 }
2060 
2061 
2062 #define CV_FONT_SIZE_SHIFT     8
2063 #define CV_FONT_ITALIC_ALPHA   (1 << 8)
2064 #define CV_FONT_ITALIC_DIGIT   (2 << 8)
2065 #define CV_FONT_ITALIC_PUNCT   (4 << 8)
2066 #define CV_FONT_ITALIC_BRACES  (8 << 8)
2067 #define CV_FONT_HAVE_GREEK     (16 << 8)
2068 #define CV_FONT_HAVE_CYRILLIC  (32 << 8)
2069 
2070 static const int icvHersheyPlain[] = {
2071 (5 + 4*16) + CV_FONT_HAVE_GREEK,
2072 199, 214, 217, 233, 219, 197, 234, 216, 221, 222, 228, 225, 211, 224, 210, 220,
2073 200, 201, 202, 203, 204, 205, 206, 207, 208, 209, 212, 213, 191, 226, 192,
2074 215, 190, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13,
2075 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 193, 84,
2076 194, 85, 86, 87, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111,
2077 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126,
2078 195, 223, 196, 88 };
2079 
2080 static const int icvHersheyPlainItalic[] = {
2081 (5 + 4*16) + CV_FONT_ITALIC_ALPHA + CV_FONT_HAVE_GREEK,
2082 199, 214, 217, 233, 219, 197, 234, 216, 221, 222, 228, 225, 211, 224, 210, 220,
2083 200, 201, 202, 203, 204, 205, 206, 207, 208, 209, 212, 213, 191, 226, 192,
2084 215, 190, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63,
2085 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 193, 84,
2086 194, 85, 86, 87, 151, 152, 153, 154, 155, 156, 157, 158, 159, 160, 161,
2087 162, 163, 164, 165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175, 176,
2088 195, 223, 196, 88 };
2089 
2090 static const int icvHersheyComplexSmall[] = {
2091 (6 + 7*16) + CV_FONT_HAVE_GREEK,
2092 1199, 1214, 1217, 1275, 1274, 1271, 1272, 1216, 1221, 1222, 1219, 1232, 1211, 1231, 1210, 1220,
2093 1200, 1201, 1202, 1203, 1204, 1205, 1206, 1207, 1208, 1209, 1212, 2213, 1241, 1238, 1242,
2094 1215, 1273, 1001, 1002, 1003, 1004, 1005, 1006, 1007, 1008, 1009, 1010, 1011, 1012, 1013,
2095 1014, 1015, 1016, 1017, 1018, 1019, 1020, 1021, 1022, 1023, 1024, 1025, 1026, 1223, 1084,
2096 1224, 1247, 586, 1249, 1101, 1102, 1103, 1104, 1105, 1106, 1107, 1108, 1109, 1110, 1111,
2097 1112, 1113, 1114, 1115, 1116, 1117, 1118, 1119, 1120, 1121, 1122, 1123, 1124, 1125, 1126,
2098 1225, 1229, 1226, 1246 };
2099 
2100 static const int icvHersheyComplexSmallItalic[] = {
2101 (6 + 7*16) + CV_FONT_ITALIC_ALPHA + CV_FONT_HAVE_GREEK,
2102 1199, 1214, 1217, 1275, 1274, 1271, 1272, 1216, 1221, 1222, 1219, 1232, 1211, 1231, 1210, 1220,
2103 1200, 1201, 1202, 1203, 1204, 1205, 1206, 1207, 1208, 1209, 1212, 1213, 1241, 1238, 1242,
2104 1215, 1273, 1051, 1052, 1053, 1054, 1055, 1056, 1057, 1058, 1059, 1060, 1061, 1062, 1063,
2105 1064, 1065, 1066, 1067, 1068, 1069, 1070, 1071, 1072, 1073, 1074, 1075, 1076, 1223, 1084,
2106 1224, 1247, 586, 1249, 1151, 1152, 1153, 1154, 1155, 1156, 1157, 1158, 1159, 1160, 1161,
2107 1162, 1163, 1164, 1165, 1166, 1167, 1168, 1169, 1170, 1171, 1172, 1173, 1174, 1175, 1176,
2108 1225, 1229, 1226, 1246 };
2109 
2110 static const int icvHersheySimplex[] = {
2111 (9 + 12*16) + CV_FONT_HAVE_GREEK,
2112 2199, 714, 717, 733, 719, 697, 734, 716, 721, 722, 728, 725, 711, 724, 710, 720,
2113 700, 701, 702, 703, 704, 705, 706, 707, 708, 709, 712, 713, 691, 726, 692,
2114 715, 690, 501, 502, 503, 504, 505, 506, 507, 508, 509, 510, 511, 512, 513,
2115 514, 515, 516, 517, 518, 519, 520, 521, 522, 523, 524, 525, 526, 693, 584,
2116 694, 2247, 586, 2249, 601, 602, 603, 604, 605, 606, 607, 608, 609, 610, 611,
2117 612, 613, 614, 615, 616, 617, 618, 619, 620, 621, 622, 623, 624, 625, 626,
2118 695, 723, 696, 2246 };
2119 
2120 static const int icvHersheyDuplex[] = {
2121 (9 + 12*16) + CV_FONT_HAVE_GREEK,
2122 2199, 2714, 2728, 2732, 2719, 2733, 2718, 2727, 2721, 2722, 2723, 2725, 2711, 2724, 2710, 2720,
2123 2700, 2701, 2702, 2703, 2704, 2705, 2706, 2707, 2708, 2709, 2712, 2713, 2730, 2726, 2731,
2124 2715, 2734, 2501, 2502, 2503, 2504, 2505, 2506, 2507, 2508, 2509, 2510, 2511, 2512, 2513,
2125 2514, 2515, 2516, 2517, 2518, 2519, 2520, 2521, 2522, 2523, 2524, 2525, 2526, 2223, 2084,
2126 2224, 2247, 587, 2249, 2601, 2602, 2603, 2604, 2605, 2606, 2607, 2608, 2609, 2610, 2611,
2127 2612, 2613, 2614, 2615, 2616, 2617, 2618, 2619, 2620, 2621, 2622, 2623, 2624, 2625, 2626,
2128 2225, 2229, 2226, 2246 };
2129 
2130 static const int icvHersheyComplex[] = {
2131 (9 + 12*16) + CV_FONT_HAVE_GREEK + CV_FONT_HAVE_CYRILLIC,
2132 2199, 2214, 2217, 2275, 2274, 2271, 2272, 2216, 2221, 2222, 2219, 2232, 2211, 2231, 2210, 2220,
2133 2200, 2201, 2202, 2203, 2204, 2205, 2206, 2207, 2208, 2209, 2212, 2213, 2241, 2238, 2242,
2134 2215, 2273, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013,
2135 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021, 2022, 2023, 2024, 2025, 2026, 2223, 2084,
2136 2224, 2247, 587, 2249, 2101, 2102, 2103, 2104, 2105, 2106, 2107, 2108, 2109, 2110, 2111,
2137 2112, 2113, 2114, 2115, 2116, 2117, 2118, 2119, 2120, 2121, 2122, 2123, 2124, 2125, 2126,
2138 2225, 2229, 2226, 2246 };
2139 
2140 static const int icvHersheyComplexItalic[] = {
2141 (9 + 12*16) + CV_FONT_ITALIC_ALPHA + CV_FONT_ITALIC_DIGIT + CV_FONT_ITALIC_PUNCT +
2142 CV_FONT_HAVE_GREEK + CV_FONT_HAVE_CYRILLIC,
2143 2199, 2764, 2778, 2782, 2769, 2783, 2768, 2777, 2771, 2772, 2219, 2232, 2211, 2231, 2210, 2220,
2144 2750, 2751, 2752, 2753, 2754, 2755, 2756, 2757, 2758, 2759, 2212, 2213, 2241, 2238, 2242,
2145 2765, 2273, 2051, 2052, 2053, 2054, 2055, 2056, 2057, 2058, 2059, 2060, 2061, 2062, 2063,
2146 2064, 2065, 2066, 2067, 2068, 2069, 2070, 2071, 2072, 2073, 2074, 2075, 2076, 2223, 2084,
2147 2224, 2247, 587, 2249, 2151, 2152, 2153, 2154, 2155, 2156, 2157, 2158, 2159, 2160, 2161,
2148 2162, 2163, 2164, 2165, 2166, 2167, 2168, 2169, 2170, 2171, 2172, 2173, 2174, 2175, 2176,
2149 2225, 2229, 2226, 2246 };
2150 
2151 static const int icvHersheyTriplex[] = {
2152 (9 + 12*16) + CV_FONT_HAVE_GREEK,
2153 2199, 3214, 3228, 3232, 3219, 3233, 3218, 3227, 3221, 3222, 3223, 3225, 3211, 3224, 3210, 3220,
2154 3200, 3201, 3202, 3203, 3204, 3205, 3206, 3207, 3208, 3209, 3212, 3213, 3230, 3226, 3231,
2155 3215, 3234, 3001, 3002, 3003, 3004, 3005, 3006, 3007, 3008, 3009, 3010, 3011, 3012, 3013,
2156 2014, 3015, 3016, 3017, 3018, 3019, 3020, 3021, 3022, 3023, 3024, 3025, 3026, 2223, 2084,
2157 2224, 2247, 587, 2249, 3101, 3102, 3103, 3104, 3105, 3106, 3107, 3108, 3109, 3110, 3111,
2158 3112, 3113, 3114, 3115, 3116, 3117, 3118, 3119, 3120, 3121, 3122, 3123, 3124, 3125, 3126,
2159 2225, 2229, 2226, 2246 };
2160 
2161 static const int icvHersheyTriplexItalic[] = {
2162 (9 + 12*16) + CV_FONT_ITALIC_ALPHA + CV_FONT_ITALIC_DIGIT +
2163 CV_FONT_ITALIC_PUNCT + CV_FONT_HAVE_GREEK,
2164 2199, 3264, 3278, 3282, 3269, 3233, 3268, 3277, 3271, 3272, 3223, 3225, 3261, 3224, 3260, 3270,
2165 3250, 3251, 3252, 3253, 3254, 3255, 3256, 3257, 3258, 3259, 3262, 3263, 3230, 3226, 3231,
2166 3265, 3234, 3051, 3052, 3053, 3054, 3055, 3056, 3057, 3058, 3059, 3060, 3061, 3062, 3063,
2167 2064, 3065, 3066, 3067, 3068, 3069, 3070, 3071, 3072, 3073, 3074, 3075, 3076, 2223, 2084,
2168 2224, 2247, 587, 2249, 3151, 3152, 3153, 3154, 3155, 3156, 3157, 3158, 3159, 3160, 3161,
2169 3162, 3163, 3164, 3165, 3166, 3167, 3168, 3169, 3170, 3171, 3172, 3173, 3174, 3175, 3176,
2170 2225, 2229, 2226, 2246 };
2171 
2172 static const int icvHersheyScriptSimplex[] = {
2173 (9 + 12*16) + CV_FONT_ITALIC_ALPHA + CV_FONT_HAVE_GREEK,
2174 2199, 714, 717, 733, 719, 697, 734, 716, 721, 722, 728, 725, 711, 724, 710, 720,
2175 700, 701, 702, 703, 704, 705, 706, 707, 708, 709, 712, 713, 691, 726, 692,
2176 715, 690, 551, 552, 553, 554, 555, 556, 557, 558, 559, 560, 561, 562, 563,
2177 564, 565, 566, 567, 568, 569, 570, 571, 572, 573, 574, 575, 576, 693, 584,
2178 694, 2247, 586, 2249, 651, 652, 653, 654, 655, 656, 657, 658, 659, 660, 661,
2179 662, 663, 664, 665, 666, 667, 668, 669, 670, 671, 672, 673, 674, 675, 676,
2180 695, 723, 696, 2246 };
2181 
2182 static const int icvHersheyScriptComplex[] = {
2183 (9 + 12*16) + CV_FONT_ITALIC_ALPHA + CV_FONT_ITALIC_DIGIT + CV_FONT_ITALIC_PUNCT + CV_FONT_HAVE_GREEK,
2184 2199, 2764, 2778, 2782, 2769, 2783, 2768, 2777, 2771, 2772, 2219, 2232, 2211, 2231, 2210, 2220,
2185 2750, 2751, 2752, 2753, 2754, 2755, 2756, 2757, 2758, 2759, 2212, 2213, 2241, 2238, 2242,
2186 2215, 2273, 2551, 2552, 2553, 2554, 2555, 2556, 2557, 2558, 2559, 2560, 2561, 2562, 2563,
2187 2564, 2565, 2566, 2567, 2568, 2569, 2570, 2571, 2572, 2573, 2574, 2575, 2576, 2223, 2084,
2188 2224, 2247, 586, 2249, 2651, 2652, 2653, 2654, 2655, 2656, 2657, 2658, 2659, 2660, 2661,
2189 2662, 2663, 2664, 2665, 2666, 2667, 2668, 2669, 2670, 2671, 2672, 2673, 2674, 2675, 2676,
2190 2225, 2229, 2226, 2246 };
2191 
2192 
2193 CV_IMPL void
cvPutText(void * img,const char * text,CvPoint org,const CvFont * font,CvScalar color)2194 cvPutText( void *img, const char *text, CvPoint org, const CvFont *font, CvScalar color )
2195 {
2196     CV_FUNCNAME( "cvPutText" );
2197 
2198     __BEGIN__;
2199 
2200     int view_x, view_y;
2201     int coi = 0;
2202     int top_bottom = 0, base_line;
2203     int hscale, vscale, default_shear, italic_shear;
2204     int thickness, line_type;
2205     CvMat stub, *mat = (CvMat*)img;
2206     double buf[4];
2207     CvPoint pt[1 << 10];
2208     int count;
2209 
2210     int i;
2211     const char **faces = icvHersheyGlyphs;
2212 
2213     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
2214 
2215     if( coi != 0 )
2216         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
2217 
2218     if( CV_IS_IMAGE_HDR(img) && ((IplImage*)img)->origin )
2219         top_bottom = 1;
2220 
2221     if( !text || !font || !font->ascii )
2222         CV_ERROR( CV_StsNullPtr, "" );
2223 
2224     CV_CALL( cvScalarToRawData( &color, buf, mat->type, 0 ));
2225     base_line = -(font->ascii[0] & 15);
2226     hscale = cvRound(font->hscale*XY_ONE);
2227     vscale = cvRound(font->vscale*XY_ONE);
2228     default_shear = cvRound(font->shear*font->vscale*XY_ONE);
2229     italic_shear = !(font->font_face & CV_FONT_ITALIC) ? 0 : cvRound(font->vscale*.25*XY_ONE);
2230     thickness = font->thickness;
2231     line_type = font->line_type;
2232 
2233     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
2234         line_type = 8;
2235 
2236     if( top_bottom )
2237         vscale = -vscale;
2238 
2239     view_x = org.x << XY_SHIFT;
2240     view_y = (org.y << XY_SHIFT) + base_line*vscale;
2241 
2242     for( i = 0; text[i] != '\0'; i++ )
2243     {
2244         int c = (uchar)text[i];
2245         int dx, shear = default_shear;
2246         const char* ptr;
2247         CvPoint p;
2248 
2249         if( c > 128 || c < ' ' )
2250             c = '?';
2251 
2252         if( italic_shear )
2253         {
2254             if( ('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z'))
2255             {
2256                 if( !(font->ascii[0] & CV_FONT_ITALIC_ALPHA) )
2257                     shear += italic_shear;
2258             }
2259             else if( '0' <= c && c <= '9' )
2260             {
2261                 if( !(font->ascii[0] & CV_FONT_ITALIC_DIGIT) )
2262                     shear += italic_shear;
2263             }
2264             else if( c < 'A' )
2265             {
2266                 if( !(font->ascii[0] & CV_FONT_ITALIC_PUNCT) )
2267                     shear += italic_shear;
2268             }
2269             else
2270             {
2271                 shear += italic_shear;
2272             }
2273         }
2274 
2275         ptr = faces[font->ascii[(c-' ')+1]];
2276         p.x = (unsigned char)ptr[0] - 'R';
2277         p.y = (unsigned char)ptr[1] - 'R';
2278         dx = p.y*hscale;
2279         view_x -= p.x*hscale;
2280         count = 0;
2281 
2282         for( ptr += 2;; )
2283         {
2284             if( *ptr == ' ' || !*ptr )
2285             {
2286                 if( count > 1 )
2287                     icvPolyLine( mat, pt, count, 0, buf, thickness, line_type, XY_SHIFT );
2288                 if( !*ptr++ )
2289                     break;
2290                 count = 0;
2291             }
2292             else
2293             {
2294                 p.x = (unsigned char)ptr[0] - 'R';
2295                 p.y = (unsigned char)ptr[1] - 'R';
2296                 ptr += 2;
2297                 pt[count].x = p.x*hscale - p.y*shear + view_x;
2298                 pt[count++].y = p.y*vscale + view_y;
2299             }
2300         }
2301         view_x += dx;
2302     }
2303 
2304     __END__;
2305 }
2306 
2307 CV_IMPL void
cvInitFont(CvFont * font,int font_face,double hscale,double vscale,double shear,int thickness,int line_type)2308 cvInitFont( CvFont *font, int font_face, double hscale, double vscale,
2309             double shear, int thickness, int line_type )
2310 {
2311     CV_FUNCNAME( "cvInitFont" );
2312 
2313     __BEGIN__;
2314 
2315     int is_italic = font_face & CV_FONT_ITALIC;
2316 
2317     if( !font )
2318         CV_ERROR( CV_StsNullPtr, "" );
2319 
2320     if( hscale <= 0 || vscale <= 0 || thickness < 0 )
2321         CV_ERROR( CV_StsOutOfRange, "" );
2322 
2323     switch( (font_face & 7) )
2324     {
2325     case CV_FONT_HERSHEY_SIMPLEX:
2326         font->ascii = icvHersheySimplex;
2327         break;
2328     case CV_FONT_HERSHEY_PLAIN:
2329         font->ascii = !is_italic ? icvHersheyPlain : icvHersheyPlainItalic;
2330         break;
2331     case CV_FONT_HERSHEY_DUPLEX:
2332         font->ascii = icvHersheyDuplex;
2333         break;
2334     case CV_FONT_HERSHEY_COMPLEX:
2335         font->ascii = !is_italic ? icvHersheyComplex : icvHersheyComplexItalic;
2336         break;
2337     case CV_FONT_HERSHEY_TRIPLEX:
2338         font->ascii = !is_italic ? icvHersheyTriplex : icvHersheyTriplexItalic;
2339         break;
2340     case CV_FONT_HERSHEY_COMPLEX_SMALL:
2341         font->ascii = !is_italic ? icvHersheyComplexSmall : icvHersheyComplexSmallItalic;
2342         break;
2343     case CV_FONT_HERSHEY_SCRIPT_SIMPLEX:
2344         font->ascii = icvHersheyScriptSimplex;
2345         break;
2346     case CV_FONT_HERSHEY_SCRIPT_COMPLEX:
2347         font->ascii = icvHersheyScriptComplex;
2348         break;
2349     default:
2350         CV_ERROR( CV_StsOutOfRange, "Unknown font type" );
2351     }
2352 
2353     font->font_face = font_face;
2354     font->hscale = (float)hscale;
2355     font->vscale = (float)vscale;
2356     font->thickness = thickness;
2357     font->shear = (float)shear;
2358     font->greek = font->cyrillic = 0;
2359     font->line_type = line_type;
2360 
2361     __END__;
2362 }
2363 
2364 
2365 CV_IMPL void
cvGetTextSize(const char * text,const CvFont * font,CvSize * size,int * _base_line)2366 cvGetTextSize( const char *text, const CvFont *font, CvSize *size, int *_base_line )
2367 {
2368     CV_FUNCNAME( "cvGetTextSize" );
2369 
2370     __BEGIN__;
2371 
2372     float view_x = 0;
2373     int base_line, cap_line;
2374 
2375     int i;
2376     const char **faces = icvHersheyGlyphs;
2377 
2378     if( !text || !font || !font->ascii || !size )
2379         CV_ERROR( CV_StsNullPtr, "" );
2380 
2381     base_line = (font->ascii[0] & 15);
2382     cap_line = (font->ascii[0] >> 4) & 15;
2383     if( _base_line )
2384         *_base_line = cvRound(base_line*font->vscale);
2385     size->height = cvRound((cap_line + base_line)*font->vscale + font->thickness);
2386 
2387     for( i = 0; text[i] != '\0'; i++ )
2388     {
2389         int c = (uchar)text[i];
2390         const char* ptr;
2391         CvPoint p;
2392 
2393         if( c > 128 || c < ' ' )
2394             c = '?';
2395 
2396         ptr = faces[font->ascii[(c-' ')+1]];
2397         p.x = (unsigned char)ptr[0] - 'R';
2398         p.y = (unsigned char)ptr[1] - 'R';
2399         view_x += (p.y - p.x)*font->hscale;
2400     }
2401 
2402     size->width = cvRound(view_x + font->thickness);
2403 
2404     __END__;
2405 }
2406 
2407 
2408 static const CvPoint icvCodeDeltas[8] =
2409 { {1, 0}, {1, -1}, {0, -1}, {-1, -1}, {-1, 0}, {-1, 1}, {0, 1}, {1, 1} };
2410 
2411 #define CV_ADJUST_EDGE_COUNT( count, seq )  \
2412     ((count) -= ((count) == (seq)->total && !CV_IS_SEQ_CLOSED(seq)))
2413 
2414 CV_IMPL void
cvDrawContours(void * img,CvSeq * contour,CvScalar externalColor,CvScalar holeColor,int maxLevel,int thickness,int line_type,CvPoint offset)2415 cvDrawContours( void*  img,  CvSeq*  contour,
2416                 CvScalar externalColor, CvScalar holeColor,
2417                 int  maxLevel, int thickness,
2418                 int line_type, CvPoint offset )
2419 {
2420     CvSeq *contour0 = contour, *h_next = 0;
2421     CvMemStorage* st = 0;
2422     CvSeq* tseq = 0;
2423     CvContour* edges = 0;
2424     CvSeqWriter writer;
2425     CvTreeNodeIterator iterator;
2426 
2427     CV_FUNCNAME( "cvDrawContours" );
2428 
2429     __BEGIN__;
2430 
2431     int coi = 0;
2432     CvMat stub, *mat = (CvMat*)img;
2433     double ext_buf[4], hole_buf[4];
2434 
2435     CV_CALL( mat = cvGetMat( mat, &stub, &coi ));
2436 
2437     if( line_type == CV_AA && CV_MAT_DEPTH(mat->type) != CV_8U )
2438         line_type = 8;
2439 
2440     if( !contour )
2441         EXIT;
2442 
2443     if( coi != 0 )
2444         CV_ERROR( CV_BadCOI, cvUnsupportedFormat );
2445 
2446     if( thickness < -1 || thickness > 255 )
2447         CV_ERROR( CV_StsOutOfRange, "" );
2448 
2449     CV_CALL( cvScalarToRawData( &externalColor, ext_buf, mat->type, 0 ));
2450     CV_CALL( cvScalarToRawData( &holeColor, hole_buf, mat->type, 0 ));
2451 
2452     if( maxLevel < 0 )
2453     {
2454         h_next = contour->h_next;
2455         contour->h_next = 0;
2456         maxLevel = -maxLevel+1;
2457     }
2458 
2459     if( thickness < 0 )
2460     {
2461         if( contour->storage )
2462             st = cvCreateChildMemStorage( contour->storage );
2463         else
2464             st = cvCreateMemStorage( CV_DRAWING_STORAGE_BLOCK );
2465         tseq = cvCreateSeq( 0, sizeof(CvContour), sizeof(CvPoint), st );
2466         edges = (CvContour*)cvCreateSeq( 0, sizeof(CvContour), sizeof(CvPolyEdge), st );
2467     }
2468 
2469     memset( &writer, 0, sizeof(writer));
2470 
2471     cvInitTreeNodeIterator( &iterator, contour, maxLevel );
2472 
2473     while( (contour = (CvSeq*)cvNextTreeNode( &iterator )) != 0 )
2474     {
2475         CvSeqReader reader;
2476         int i, count = contour->total;
2477         int elem_type = CV_MAT_TYPE(contour->flags);
2478         void* clr = (contour->flags & CV_SEQ_FLAG_HOLE) == 0 ? ext_buf : hole_buf;
2479 
2480         cvStartReadSeq( contour, &reader, 0 );
2481 
2482         if( CV_IS_SEQ_CHAIN_CONTOUR( contour ))
2483         {
2484             CvPoint pt = ((CvChain*)contour)->origin;
2485             CvPoint prev_pt = pt;
2486             char prev_code = reader.ptr ? reader.ptr[0] : '\0';
2487 
2488             if( thickness < 0 )
2489             {
2490                 cvClearSeq( tseq );
2491                 cvStartAppendToSeq( tseq, &writer );
2492                 CV_WRITE_SEQ_ELEM( pt, writer );
2493             }
2494 
2495             prev_pt.x += offset.x;
2496             prev_pt.y += offset.y;
2497 
2498             for( i = 0; i < count; i++ )
2499             {
2500                 char code;
2501                 CV_READ_SEQ_ELEM( code, reader );
2502 
2503                 assert( (code & ~7) == 0 );
2504 
2505                 if( code != prev_code )
2506                 {
2507                     prev_code = code;
2508                     if( thickness >= 0 )
2509                     {
2510                         icvThickLine( mat, prev_pt, pt, clr, thickness, line_type, 2, 0 );
2511                     }
2512                     else
2513                     {
2514                         CV_WRITE_SEQ_ELEM( pt, writer );
2515                     }
2516                     prev_pt = pt;
2517                 }
2518 
2519                 pt.x += icvCodeDeltas[(int)code].x;
2520                 pt.y += icvCodeDeltas[(int)code].y;
2521             }
2522 
2523             if( thickness >= 0 )
2524             {
2525                 icvThickLine( mat, prev_pt, ((CvChain*)contour)->origin,
2526                               clr, thickness, line_type, 2, 0 );
2527             }
2528             else
2529             {
2530                 CV_WRITE_SEQ_ELEM( pt, writer );
2531                 cvEndWriteSeq( &writer );
2532                 CV_CALL( icvCollectPolyEdges( mat, tseq, edges, ext_buf, line_type, 0 ));
2533             }
2534         }
2535         else if( CV_IS_SEQ_POLYLINE( contour ))
2536         {
2537             if( thickness >= 0 )
2538             {
2539                 CvPoint pt1, pt2;
2540                 int shift = 0;
2541 
2542                 count -= !CV_IS_SEQ_CLOSED(contour);
2543                 if( elem_type == CV_32SC2 )
2544                 {
2545                     CV_READ_SEQ_ELEM( pt1, reader );
2546                     pt1.x += offset.x;
2547                     pt1.y += offset.y;
2548                 }
2549                 else
2550                 {
2551                     CvPoint2D32f pt1f;
2552                     CV_READ_SEQ_ELEM( pt1f, reader );
2553                     pt1.x = cvRound( (pt1f.x + offset.x) * XY_ONE );
2554                     pt1.y = cvRound( (pt1f.y + offset.y) * XY_ONE );
2555                     shift = XY_SHIFT;
2556                 }
2557 
2558                 for( i = 0; i < count; i++ )
2559                 {
2560                     if( elem_type == CV_32SC2 )
2561                     {
2562                         CV_READ_SEQ_ELEM( pt2, reader );
2563                         pt2.x += offset.x;
2564                         pt2.y += offset.y;
2565                     }
2566                     else
2567                     {
2568                         CvPoint2D32f pt2f;
2569                         CV_READ_SEQ_ELEM( pt2f, reader );
2570                         pt2.x = cvRound( pt2f.x * XY_ONE );
2571                         pt2.y = cvRound( pt2f.y * XY_ONE );
2572                     }
2573                     icvThickLine( mat, pt1, pt2, clr, thickness, line_type, 2, shift );
2574                     pt1 = pt2;
2575                 }
2576             }
2577             else
2578             {
2579                 CV_CALL( icvCollectPolyEdges( mat, contour, edges, ext_buf, line_type, 0, offset ));
2580             }
2581         }
2582     }
2583 
2584     if( thickness < 0 )
2585     {
2586         CV_CALL( icvFillEdgeCollection( mat, edges, ext_buf ));
2587     }
2588 
2589     __END__;
2590 
2591     if( h_next && contour0 )
2592         contour0->h_next = h_next;
2593 
2594     cvReleaseMemStorage( &st );
2595 }
2596 
2597 /* End of file. */
2598