gdiplus: Added GdipAddPathLine2.
[wine] / dlls / gdiplus / graphicspath.c
1 /*
2  * Copyright (C) 2007 Google (Evan Stade)
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  *
18  */
19
20 #include <stdarg.h>
21 #include <math.h>
22
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winuser.h"
26 #include "wingdi.h"
27 #include "gdiplus.h"
28 #include "gdiplus_private.h"
29 #include "wine/debug.h"
30
31 WINE_DEFAULT_DEBUG_CHANNEL(gdiplus);
32
33 /* make sure path has enough space for len more points */
34 static BOOL lengthen_path(GpPath *path, INT len)
35 {
36     /* initial allocation */
37     if(path->datalen == 0){
38         path->datalen = len * 2;
39
40         path->pathdata.Points = GdipAlloc(path->datalen * sizeof(PointF));
41         if(!path->pathdata.Points)   return FALSE;
42
43         path->pathdata.Types = GdipAlloc(path->datalen);
44         if(!path->pathdata.Types){
45             GdipFree(path->pathdata.Points);
46             return FALSE;
47         }
48     }
49     /* reallocation, double size of arrays */
50     else if(path->datalen - path->pathdata.Count < len){
51         while(path->datalen - path->pathdata.Count < len)
52             path->datalen *= 2;
53
54         path->pathdata.Points = HeapReAlloc(GetProcessHeap(), 0,
55             path->pathdata.Points, path->datalen * sizeof(PointF));
56         if(!path->pathdata.Points)  return FALSE;
57
58         path->pathdata.Types = HeapReAlloc(GetProcessHeap(), 0,
59             path->pathdata.Types, path->datalen);
60         if(!path->pathdata.Types)   return FALSE;
61     }
62
63     return TRUE;
64 }
65
66 GpStatus WINGDIPAPI GdipAddPathLine2(GpPath *path, GDIPCONST GpPointF *points,
67     INT count)
68 {
69     INT i, old_count = path->pathdata.Count;
70
71     if(!path || !points)
72         return InvalidParameter;
73
74     if(!lengthen_path(path, count + (path->newfigure ? 1 : 0)))
75         return OutOfMemory;
76
77     for(i = 0; i < count; i++){
78         path->pathdata.Points[old_count + i].X = points[i].X;
79         path->pathdata.Points[old_count + i].Y = points[i].Y;
80         path->pathdata.Types[old_count + i] = PathPointTypeLine;
81     }
82
83     if(path->newfigure){
84         path->pathdata.Types[old_count] = PathPointTypeStart;
85         path->newfigure = FALSE;
86     }
87
88     path->pathdata.Count += count;
89
90     return Ok;
91 }
92
93 GpStatus WINGDIPAPI GdipCreatePath(GpFillMode fill, GpPath **path)
94 {
95     if(!path)
96         return InvalidParameter;
97
98     *path = GdipAlloc(sizeof(GpPath));
99     if(!*path)  return OutOfMemory;
100
101     (*path)->fill = fill;
102     (*path)->newfigure = TRUE;
103
104     return Ok;
105 }
106
107 GpStatus WINGDIPAPI GdipDeletePath(GpPath *path)
108 {
109     if(!path)
110         return InvalidParameter;
111
112     GdipFree(path);
113
114     return Ok;
115 }