2 * Copyright (C) 2007 Google (Evan Stade)
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.
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.
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
28 #include "gdiplus_private.h"
29 #include "wine/debug.h"
31 WINE_DEFAULT_DEBUG_CHANNEL(gdiplus
);
33 static DWORD
gdip_to_gdi_dash(GpDashStyle dash
)
42 case DashStyleDashDot
:
44 case DashStyleDashDotDot
:
49 ERR("Not a member of GpDashStyle enumeration\n");
54 static DWORD
gdip_to_gdi_join(GpLineJoin join
)
62 case LineJoinMiterClipped
:
65 ERR("Not a member of GpLineJoin enumeration\n");
70 GpStatus WINGDIPAPI
GdipClonePen(GpPen
*pen
, GpPen
**clonepen
)
73 return InvalidParameter
;
75 *clonepen
= GdipAlloc(sizeof(GpPen
));
76 if(!*clonepen
) return OutOfMemory
;
80 GdipCloneCustomLineCap(pen
->customstart
, &(*clonepen
)->customstart
);
81 GdipCloneCustomLineCap(pen
->customend
, &(*clonepen
)->customend
);
82 GdipCloneBrush(pen
->brush
, &(*clonepen
)->brush
);
87 GpStatus WINGDIPAPI
GdipCreatePen1(ARGB color
, REAL width
, GpUnit unit
,
93 GdipCreateSolidFill(color
, (GpSolidFill
**)(&brush
));
94 status
= GdipCreatePen2(brush
, width
, unit
, pen
);
95 GdipDeleteBrush(brush
);
99 GpStatus WINGDIPAPI
GdipCreatePen2(GpBrush
*brush
, REAL width
, GpUnit unit
,
103 GpBrush
*clone_brush
;
106 return InvalidParameter
;
108 gp_pen
= GdipAlloc(sizeof(GpPen
));
109 if(!gp_pen
) return OutOfMemory
;
111 gp_pen
->style
= GP_DEFAULT_PENSTYLE
;
112 gp_pen
->width
= width
;
114 gp_pen
->endcap
= LineCapFlat
;
115 gp_pen
->join
= LineJoinMiter
;
116 gp_pen
->miterlimit
= 10.0;
117 gp_pen
->dash
= DashStyleSolid
;
118 gp_pen
->offset
= 0.0;
119 gp_pen
->customstart
= NULL
;
120 gp_pen
->customend
= NULL
;
122 if(!((gp_pen
->unit
== UnitWorld
) || (gp_pen
->unit
== UnitPixel
))) {
123 FIXME("UnitWorld, UnitPixel only supported units\n");
125 return NotImplemented
;
128 GdipCloneBrush(brush
, &clone_brush
);
129 gp_pen
->brush
= clone_brush
;
136 GpStatus WINGDIPAPI
GdipDeletePen(GpPen
*pen
)
138 if(!pen
) return InvalidParameter
;
140 GdipDeleteBrush(pen
->brush
);
141 GdipDeleteCustomLineCap(pen
->customstart
);
142 GdipDeleteCustomLineCap(pen
->customend
);
143 GdipFree(pen
->dashes
);
149 GpStatus WINGDIPAPI
GdipGetPenBrushFill(GpPen
*pen
, GpBrush
**brush
)
152 return InvalidParameter
;
154 return GdipCloneBrush(pen
->brush
, brush
);
157 GpStatus WINGDIPAPI
GdipGetPenColor(GpPen
*pen
, ARGB
*argb
)
160 return InvalidParameter
;
162 if(pen
->brush
->bt
!= BrushTypeSolidColor
)
163 return NotImplemented
;
165 return GdipGetSolidFillColor(((GpSolidFill
*)pen
->brush
), argb
);
168 GpStatus WINGDIPAPI
GdipGetPenCustomEndCap(GpPen
*pen
, GpCustomLineCap
** customCap
)
170 if(!pen
|| !customCap
)
171 return InvalidParameter
;
178 return GdipCloneCustomLineCap(pen
->customend
, customCap
);
181 GpStatus WINGDIPAPI
GdipGetPenCustomStartCap(GpPen
*pen
, GpCustomLineCap
** customCap
)
183 if(!pen
|| !customCap
)
184 return InvalidParameter
;
186 if(!pen
->customstart
){
191 return GdipCloneCustomLineCap(pen
->customstart
, customCap
);
194 GpStatus WINGDIPAPI
GdipGetPenDashArray(GpPen
*pen
, REAL
*dash
, INT count
)
196 if(!pen
|| !dash
|| count
> pen
->numdashes
)
197 return InvalidParameter
;
199 /* note: if you pass a negative value for count, it crashes native gdiplus. */
203 memcpy(dash
, pen
->dashes
, count
* sizeof(REAL
));
208 GpStatus WINGDIPAPI
GdipGetPenDashCap197819(GpPen
*pen
, GpDashCap
*dashCap
)
211 return InvalidParameter
;
213 *dashCap
= pen
->dashcap
;
218 GpStatus WINGDIPAPI
GdipGetPenDashCount(GpPen
*pen
, INT
*count
)
221 return InvalidParameter
;
223 *count
= pen
->numdashes
;
228 GpStatus WINGDIPAPI
GdipGetPenDashOffset(GpPen
*pen
, REAL
*offset
)
231 return InvalidParameter
;
233 *offset
= pen
->offset
;
238 GpStatus WINGDIPAPI
GdipGetPenDashStyle(GpPen
*pen
, GpDashStyle
*dash
)
241 return InvalidParameter
;
248 GpStatus WINGDIPAPI
GdipGetPenEndCap(GpPen
*pen
, GpLineCap
*endCap
)
251 return InvalidParameter
;
253 *endCap
= pen
->endcap
;
258 GpStatus WINGDIPAPI
GdipGetPenLineJoin(GpPen
*pen
, GpLineJoin
*lineJoin
)
260 if(!pen
|| !lineJoin
)
261 return InvalidParameter
;
263 *lineJoin
= pen
->join
;
268 GpStatus WINGDIPAPI
GdipGetPenMode(GpPen
*pen
, GpPenAlignment
*mode
)
271 return InvalidParameter
;
278 GpStatus WINGDIPAPI
GdipGetPenMiterLimit(GpPen
*pen
, REAL
*miterLimit
)
280 if(!pen
|| !miterLimit
)
281 return InvalidParameter
;
283 *miterLimit
= pen
->miterlimit
;
288 GpStatus WINGDIPAPI
GdipGetPenStartCap(GpPen
*pen
, GpLineCap
*startCap
)
290 if(!pen
|| !startCap
)
291 return InvalidParameter
;
293 *startCap
= pen
->startcap
;
298 GpStatus WINGDIPAPI
GdipGetPenUnit(GpPen
*pen
, GpUnit
*unit
)
301 return InvalidParameter
;
308 GpStatus WINGDIPAPI
GdipGetPenWidth(GpPen
*pen
, REAL
*width
)
311 return InvalidParameter
;
318 GpStatus WINGDIPAPI
GdipSetPenBrushFill(GpPen
*pen
, GpBrush
*brush
)
321 return InvalidParameter
;
323 GdipDeleteBrush(pen
->brush
);
324 return GdipCloneBrush(brush
, &pen
->brush
);
327 GpStatus WINGDIPAPI
GdipSetPenColor(GpPen
*pen
, ARGB argb
)
330 return InvalidParameter
;
332 if(pen
->brush
->bt
!= BrushTypeSolidColor
)
333 return NotImplemented
;
335 return GdipSetSolidFillColor(((GpSolidFill
*)pen
->brush
), argb
);
338 GpStatus WINGDIPAPI
GdipSetPenCustomEndCap(GpPen
*pen
, GpCustomLineCap
* customCap
)
340 GpCustomLineCap
* cap
;
343 /* native crashes on pen == NULL, customCap != NULL */
344 if(!customCap
) return InvalidParameter
;
346 if((ret
= GdipCloneCustomLineCap(customCap
, &cap
)) == Ok
){
347 GdipDeleteCustomLineCap(pen
->customend
);
348 pen
->endcap
= LineCapCustom
;
349 pen
->customend
= cap
;
355 GpStatus WINGDIPAPI
GdipSetPenCustomStartCap(GpPen
*pen
, GpCustomLineCap
* customCap
)
357 GpCustomLineCap
* cap
;
360 /* native crashes on pen == NULL, customCap != NULL */
361 if(!customCap
) return InvalidParameter
;
363 if((ret
= GdipCloneCustomLineCap(customCap
, &cap
)) == Ok
){
364 GdipDeleteCustomLineCap(pen
->customstart
);
365 pen
->startcap
= LineCapCustom
;
366 pen
->customstart
= cap
;
372 GpStatus WINGDIPAPI
GdipSetPenDashArray(GpPen
*pen
, GDIPCONST REAL
*dash
,
379 return InvalidParameter
;
384 for(i
= 0; i
< count
; i
++){
387 return InvalidParameter
;
390 if(sum
== 0.0 && count
)
391 return InvalidParameter
;
393 GdipFree(pen
->dashes
);
397 pen
->dashes
= GdipAlloc(count
* sizeof(REAL
));
403 GdipSetPenDashStyle(pen
, DashStyleCustom
);
404 memcpy(pen
->dashes
, dash
, count
* sizeof(REAL
));
405 pen
->numdashes
= count
;
410 GpStatus WINGDIPAPI
GdipSetPenDashCap197819(GpPen
*pen
, GpDashCap dashCap
)
413 return InvalidParameter
;
415 pen
->dashcap
= dashCap
;
420 /* FIXME: dash offset not used */
421 GpStatus WINGDIPAPI
GdipSetPenDashOffset(GpPen
*pen
, REAL offset
)
424 return InvalidParameter
;
426 pen
->offset
= offset
;
431 GpStatus WINGDIPAPI
GdipSetPenDashStyle(GpPen
*pen
, GpDashStyle dash
)
434 return InvalidParameter
;
436 if(dash
!= DashStyleCustom
){
437 GdipFree(pen
->dashes
);
443 pen
->style
&= ~(PS_ALTERNATE
| PS_SOLID
| PS_DASH
| PS_DOT
| PS_DASHDOT
|
444 PS_DASHDOTDOT
| PS_NULL
| PS_USERSTYLE
| PS_INSIDEFRAME
);
445 pen
->style
|= gdip_to_gdi_dash(dash
);
450 GpStatus WINGDIPAPI
GdipSetPenEndCap(GpPen
*pen
, GpLineCap cap
)
452 if(!pen
) return InvalidParameter
;
454 /* The old custom cap gets deleted even if the new style is LineCapCustom. */
455 GdipDeleteCustomLineCap(pen
->customend
);
456 pen
->customend
= NULL
;
462 /* FIXME: startcap, dashcap not used. */
463 GpStatus WINGDIPAPI
GdipSetPenLineCap197819(GpPen
*pen
, GpLineCap start
,
464 GpLineCap end
, GpDashCap dash
)
467 return InvalidParameter
;
469 GdipDeleteCustomLineCap(pen
->customend
);
470 GdipDeleteCustomLineCap(pen
->customstart
);
471 pen
->customend
= NULL
;
472 pen
->customstart
= NULL
;
474 pen
->startcap
= start
;
481 /* FIXME: Miter line joins behave a bit differently than they do in windows.
482 * Both kinds of miter joins clip if the angle is less than 11 degrees. */
483 GpStatus WINGDIPAPI
GdipSetPenLineJoin(GpPen
*pen
, GpLineJoin join
)
485 if(!pen
) return InvalidParameter
;
488 pen
->style
&= ~(PS_JOIN_ROUND
| PS_JOIN_BEVEL
| PS_JOIN_MITER
);
489 pen
->style
|= gdip_to_gdi_join(join
);
494 GpStatus WINGDIPAPI
GdipSetPenMiterLimit(GpPen
*pen
, REAL limit
)
497 return InvalidParameter
;
499 pen
->miterlimit
= limit
;
504 GpStatus WINGDIPAPI
GdipSetPenStartCap(GpPen
*pen
, GpLineCap cap
)
506 if(!pen
) return InvalidParameter
;
508 GdipDeleteCustomLineCap(pen
->customstart
);
509 pen
->customstart
= NULL
;
515 GpStatus WINGDIPAPI
GdipSetPenWidth(GpPen
*pen
, REAL width
)
517 if(!pen
) return InvalidParameter
;
524 GpStatus WINGDIPAPI
GdipSetPenMode(GpPen
*pen
, GpPenAlignment mode
)
526 if(!pen
) return InvalidParameter
;