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 static GpPenType
bt_to_pt(GpBrushType bt
)
73 case BrushTypeSolidColor
:
74 return PenTypeSolidColor
;
75 case BrushTypeHatchFill
:
76 return PenTypeHatchFill
;
77 case BrushTypeTextureFill
:
78 return PenTypeTextureFill
;
79 case BrushTypePathGradient
:
80 return PenTypePathGradient
;
81 case BrushTypeLinearGradient
:
82 return PenTypeLinearGradient
;
84 return PenTypeUnknown
;
88 GpStatus WINGDIPAPI
GdipClonePen(GpPen
*pen
, GpPen
**clonepen
)
92 TRACE("(%p, %p)\n", pen
, clonepen
);
95 return InvalidParameter
;
97 *clonepen
= heap_alloc_zero(sizeof(GpPen
));
98 if(!*clonepen
) return OutOfMemory
;
102 (*clonepen
)->customstart
= NULL
;
103 (*clonepen
)->customend
= NULL
;
104 (*clonepen
)->brush
= NULL
;
105 (*clonepen
)->dashes
= NULL
;
107 stat
= GdipCloneBrush(pen
->brush
, &(*clonepen
)->brush
);
109 if (stat
== Ok
&& pen
->customstart
)
110 stat
= GdipCloneCustomLineCap(pen
->customstart
, &(*clonepen
)->customstart
);
112 if (stat
== Ok
&& pen
->customend
)
113 stat
= GdipCloneCustomLineCap(pen
->customend
, &(*clonepen
)->customend
);
115 if (stat
== Ok
&& pen
->dashes
)
117 (*clonepen
)->dashes
= heap_alloc_zero(pen
->numdashes
* sizeof(REAL
));
118 if ((*clonepen
)->dashes
)
119 memcpy((*clonepen
)->dashes
, pen
->dashes
, pen
->numdashes
* sizeof(REAL
));
126 GdipDeletePen(*clonepen
);
131 TRACE("<-- %p\n", *clonepen
);
136 GpStatus WINGDIPAPI
GdipCreatePen1(ARGB color
, REAL width
, GpUnit unit
,
142 TRACE("(%x, %.2f, %d, %p)\n", color
, width
, unit
, pen
);
144 GdipCreateSolidFill(color
, (GpSolidFill
**)(&brush
));
145 status
= GdipCreatePen2(brush
, width
, unit
, pen
);
146 GdipDeleteBrush(brush
);
150 GpStatus WINGDIPAPI
GdipCreatePen2(GpBrush
*brush
, REAL width
, GpUnit unit
,
154 GpBrush
*clone_brush
;
156 TRACE("(%p, %.2f, %d, %p)\n", brush
, width
, unit
, pen
);
159 return InvalidParameter
;
161 gp_pen
= heap_alloc_zero(sizeof(GpPen
));
162 if(!gp_pen
) return OutOfMemory
;
164 gp_pen
->style
= GP_DEFAULT_PENSTYLE
;
165 gp_pen
->width
= width
;
167 gp_pen
->endcap
= LineCapFlat
;
168 gp_pen
->join
= LineJoinMiter
;
169 gp_pen
->miterlimit
= 10.0;
170 gp_pen
->dash
= DashStyleSolid
;
171 gp_pen
->offset
= 0.0;
172 gp_pen
->customstart
= NULL
;
173 gp_pen
->customend
= NULL
;
174 GdipSetMatrixElements(&gp_pen
->transform
, 1.0, 0.0, 0.0, 1.0, 0.0, 0.0);
176 if(!((gp_pen
->unit
== UnitWorld
) || (gp_pen
->unit
== UnitPixel
))) {
177 FIXME("UnitWorld, UnitPixel only supported units\n");
179 return NotImplemented
;
182 GdipCloneBrush(brush
, &clone_brush
);
183 gp_pen
->brush
= clone_brush
;
187 TRACE("<-- %p\n", *pen
);
192 GpStatus WINGDIPAPI
GdipDeletePen(GpPen
*pen
)
194 TRACE("(%p)\n", pen
);
196 if(!pen
) return InvalidParameter
;
198 GdipDeleteBrush(pen
->brush
);
199 GdipDeleteCustomLineCap(pen
->customstart
);
200 GdipDeleteCustomLineCap(pen
->customend
);
201 heap_free(pen
->dashes
);
207 GpStatus WINGDIPAPI
GdipGetPenBrushFill(GpPen
*pen
, GpBrush
**brush
)
209 TRACE("(%p, %p)\n", pen
, brush
);
212 return InvalidParameter
;
214 return GdipCloneBrush(pen
->brush
, brush
);
217 GpStatus WINGDIPAPI
GdipGetPenColor(GpPen
*pen
, ARGB
*argb
)
219 TRACE("(%p, %p)\n", pen
, argb
);
222 return InvalidParameter
;
224 if(pen
->brush
->bt
!= BrushTypeSolidColor
)
225 return NotImplemented
;
227 return GdipGetSolidFillColor(((GpSolidFill
*)pen
->brush
), argb
);
230 GpStatus WINGDIPAPI
GdipGetPenCustomEndCap(GpPen
*pen
, GpCustomLineCap
** customCap
)
232 TRACE("(%p, %p)\n", pen
, customCap
);
234 if(!pen
|| !customCap
)
235 return InvalidParameter
;
242 return GdipCloneCustomLineCap(pen
->customend
, customCap
);
245 GpStatus WINGDIPAPI
GdipGetPenCustomStartCap(GpPen
*pen
, GpCustomLineCap
** customCap
)
247 TRACE("(%p, %p)\n", pen
, customCap
);
249 if(!pen
|| !customCap
)
250 return InvalidParameter
;
252 if(!pen
->customstart
){
257 return GdipCloneCustomLineCap(pen
->customstart
, customCap
);
260 GpStatus WINGDIPAPI
GdipGetPenDashArray(GpPen
*pen
, REAL
*dash
, INT count
)
262 TRACE("(%p, %p, %d)\n", pen
, dash
, count
);
264 if(!pen
|| !dash
|| count
> pen
->numdashes
)
265 return InvalidParameter
;
267 /* note: if you pass a negative value for count, it crashes native gdiplus. */
271 memcpy(dash
, pen
->dashes
, count
* sizeof(REAL
));
276 GpStatus WINGDIPAPI
GdipGetPenDashCap197819(GpPen
*pen
, GpDashCap
*dashCap
)
278 TRACE("(%p, %p)\n", pen
, dashCap
);
281 return InvalidParameter
;
283 *dashCap
= pen
->dashcap
;
288 GpStatus WINGDIPAPI
GdipGetPenDashCount(GpPen
*pen
, INT
*count
)
290 TRACE("(%p, %p)\n", pen
, count
);
293 return InvalidParameter
;
295 *count
= pen
->numdashes
;
300 GpStatus WINGDIPAPI
GdipGetPenDashOffset(GpPen
*pen
, REAL
*offset
)
302 TRACE("(%p, %p)\n", pen
, offset
);
305 return InvalidParameter
;
307 *offset
= pen
->offset
;
312 GpStatus WINGDIPAPI
GdipGetPenDashStyle(GpPen
*pen
, GpDashStyle
*dash
)
314 TRACE("(%p, %p)\n", pen
, dash
);
317 return InvalidParameter
;
324 GpStatus WINGDIPAPI
GdipGetPenEndCap(GpPen
*pen
, GpLineCap
*endCap
)
326 TRACE("(%p, %p)\n", pen
, endCap
);
329 return InvalidParameter
;
331 *endCap
= pen
->endcap
;
336 GpStatus WINGDIPAPI
GdipGetPenFillType(GpPen
*pen
, GpPenType
* type
)
338 TRACE("(%p, %p)\n", pen
, type
);
341 return InvalidParameter
;
343 *type
= bt_to_pt(pen
->brush
->bt
);
348 GpStatus WINGDIPAPI
GdipGetPenLineJoin(GpPen
*pen
, GpLineJoin
*lineJoin
)
350 TRACE("(%p, %p)\n", pen
, lineJoin
);
352 if(!pen
|| !lineJoin
)
353 return InvalidParameter
;
355 *lineJoin
= pen
->join
;
360 GpStatus WINGDIPAPI
GdipGetPenMode(GpPen
*pen
, GpPenAlignment
*mode
)
362 TRACE("(%p, %p)\n", pen
, mode
);
365 return InvalidParameter
;
372 GpStatus WINGDIPAPI
GdipGetPenMiterLimit(GpPen
*pen
, REAL
*miterLimit
)
374 TRACE("(%p, %p)\n", pen
, miterLimit
);
376 if(!pen
|| !miterLimit
)
377 return InvalidParameter
;
379 *miterLimit
= pen
->miterlimit
;
384 GpStatus WINGDIPAPI
GdipGetPenStartCap(GpPen
*pen
, GpLineCap
*startCap
)
386 TRACE("(%p, %p)\n", pen
, startCap
);
388 if(!pen
|| !startCap
)
389 return InvalidParameter
;
391 *startCap
= pen
->startcap
;
396 GpStatus WINGDIPAPI
GdipGetPenUnit(GpPen
*pen
, GpUnit
*unit
)
398 TRACE("(%p, %p)\n", pen
, unit
);
401 return InvalidParameter
;
408 GpStatus WINGDIPAPI
GdipGetPenWidth(GpPen
*pen
, REAL
*width
)
410 TRACE("(%p, %p)\n", pen
, width
);
413 return InvalidParameter
;
420 GpStatus WINGDIPAPI
GdipResetPenTransform(GpPen
*pen
)
424 TRACE("(%p)\n", pen
);
427 return InvalidParameter
;
430 FIXME("(%p) stub\n", pen
);
432 return NotImplemented
;
435 GpStatus WINGDIPAPI
GdipSetPenTransform(GpPen
*pen
, GpMatrix
*matrix
)
439 TRACE("(%p,%p)\n", pen
, matrix
);
442 return InvalidParameter
;
445 FIXME("not implemented\n");
447 return NotImplemented
;
450 GpStatus WINGDIPAPI
GdipGetPenTransform(GpPen
*pen
, GpMatrix
*matrix
)
452 TRACE("(%p,%p)\n", pen
, matrix
);
455 return InvalidParameter
;
457 *matrix
= pen
->transform
;
462 GpStatus WINGDIPAPI
GdipTranslatePenTransform(GpPen
*pen
, REAL dx
, REAL dy
, GpMatrixOrder order
)
466 TRACE("(%p,%0.2f,%0.2f,%u)\n", pen
, dx
, dy
, order
);
469 return InvalidParameter
;
472 FIXME("not implemented\n");
474 return NotImplemented
;
477 GpStatus WINGDIPAPI
GdipScalePenTransform(GpPen
*pen
, REAL sx
, REAL sy
, GpMatrixOrder order
)
481 TRACE("(%p,%0.2f,%0.2f,%u)\n", pen
, sx
, sy
, order
);
484 return InvalidParameter
;
487 FIXME("(%p, %.2f, %.2f, %d) stub\n", pen
, sx
, sy
, order
);
489 return NotImplemented
;
492 GpStatus WINGDIPAPI
GdipRotatePenTransform(GpPen
*pen
, REAL angle
, GpMatrixOrder order
)
496 TRACE("(%p,%0.2f,%u)\n", pen
, angle
, order
);
499 return InvalidParameter
;
502 FIXME("not implemented\n");
504 return NotImplemented
;
507 GpStatus WINGDIPAPI
GdipMultiplyPenTransform(GpPen
*pen
, GDIPCONST GpMatrix
*matrix
,
512 TRACE("(%p,%p,%u)\n", pen
, matrix
, order
);
515 return InvalidParameter
;
518 FIXME("not implemented\n");
520 return NotImplemented
;
523 GpStatus WINGDIPAPI
GdipSetPenBrushFill(GpPen
*pen
, GpBrush
*brush
)
525 TRACE("(%p, %p)\n", pen
, brush
);
528 return InvalidParameter
;
530 GdipDeleteBrush(pen
->brush
);
531 return GdipCloneBrush(brush
, &pen
->brush
);
534 GpStatus WINGDIPAPI
GdipSetPenColor(GpPen
*pen
, ARGB argb
)
536 TRACE("(%p, %x)\n", pen
, argb
);
539 return InvalidParameter
;
541 if(pen
->brush
->bt
!= BrushTypeSolidColor
)
542 return NotImplemented
;
544 return GdipSetSolidFillColor(((GpSolidFill
*)pen
->brush
), argb
);
547 GpStatus WINGDIPAPI
GdipGetPenCompoundCount(GpPen
*pen
, INT
*count
)
549 FIXME("(%p, %p): stub\n", pen
, count
);
552 return InvalidParameter
;
554 return NotImplemented
;
557 GpStatus WINGDIPAPI
GdipSetPenCompoundArray(GpPen
*pen
, GDIPCONST REAL
*dash
,
560 FIXME("(%p, %p, %i): stub\n", pen
, dash
, count
);
562 if (!pen
|| !dash
|| count
< 2 || count
%2 == 1)
563 return InvalidParameter
;
565 return NotImplemented
;
568 GpStatus WINGDIPAPI
GdipSetPenCustomEndCap(GpPen
*pen
, GpCustomLineCap
* customCap
)
570 GpCustomLineCap
* cap
;
573 TRACE("(%p, %p)\n", pen
, customCap
);
575 /* native crashes on pen == NULL, customCap != NULL */
576 if(!customCap
) return InvalidParameter
;
578 if((ret
= GdipCloneCustomLineCap(customCap
, &cap
)) == Ok
){
579 GdipDeleteCustomLineCap(pen
->customend
);
580 pen
->endcap
= LineCapCustom
;
581 pen
->customend
= cap
;
587 GpStatus WINGDIPAPI
GdipSetPenCustomStartCap(GpPen
*pen
, GpCustomLineCap
* customCap
)
589 GpCustomLineCap
* cap
;
592 TRACE("(%p, %p)\n", pen
, customCap
);
594 /* native crashes on pen == NULL, customCap != NULL */
595 if(!customCap
) return InvalidParameter
;
597 if((ret
= GdipCloneCustomLineCap(customCap
, &cap
)) == Ok
){
598 GdipDeleteCustomLineCap(pen
->customstart
);
599 pen
->startcap
= LineCapCustom
;
600 pen
->customstart
= cap
;
606 GpStatus WINGDIPAPI
GdipSetPenDashArray(GpPen
*pen
, GDIPCONST REAL
*dash
,
612 TRACE("(%p, %p, %d)\n", pen
, dash
, count
);
615 return InvalidParameter
;
620 for(i
= 0; i
< count
; i
++){
623 return InvalidParameter
;
626 if(sum
== 0.0 && count
)
627 return InvalidParameter
;
629 heap_free(pen
->dashes
);
633 pen
->dashes
= heap_alloc_zero(count
* sizeof(REAL
));
639 GdipSetPenDashStyle(pen
, DashStyleCustom
);
640 memcpy(pen
->dashes
, dash
, count
* sizeof(REAL
));
641 pen
->numdashes
= count
;
646 GpStatus WINGDIPAPI
GdipSetPenDashCap197819(GpPen
*pen
, GpDashCap dashCap
)
648 TRACE("(%p, %d)\n", pen
, dashCap
);
651 return InvalidParameter
;
653 pen
->dashcap
= dashCap
;
658 /* FIXME: dash offset not used */
659 GpStatus WINGDIPAPI
GdipSetPenDashOffset(GpPen
*pen
, REAL offset
)
661 TRACE("(%p, %.2f)\n", pen
, offset
);
664 return InvalidParameter
;
666 pen
->offset
= offset
;
671 GpStatus WINGDIPAPI
GdipSetPenDashStyle(GpPen
*pen
, GpDashStyle dash
)
673 TRACE("(%p, %d)\n", pen
, dash
);
676 return InvalidParameter
;
678 if(dash
!= DashStyleCustom
){
679 heap_free(pen
->dashes
);
685 pen
->style
&= ~(PS_ALTERNATE
| PS_SOLID
| PS_DASH
| PS_DOT
| PS_DASHDOT
|
686 PS_DASHDOTDOT
| PS_NULL
| PS_USERSTYLE
| PS_INSIDEFRAME
);
687 pen
->style
|= gdip_to_gdi_dash(dash
);
692 GpStatus WINGDIPAPI
GdipSetPenEndCap(GpPen
*pen
, GpLineCap cap
)
694 TRACE("(%p, %d)\n", pen
, cap
);
696 if(!pen
) return InvalidParameter
;
698 /* The old custom cap gets deleted even if the new style is LineCapCustom. */
699 GdipDeleteCustomLineCap(pen
->customend
);
700 pen
->customend
= NULL
;
706 /* FIXME: startcap, dashcap not used. */
707 GpStatus WINGDIPAPI
GdipSetPenLineCap197819(GpPen
*pen
, GpLineCap start
,
708 GpLineCap end
, GpDashCap dash
)
710 TRACE("%p, %d, %d, %d)\n", pen
, start
, end
, dash
);
713 return InvalidParameter
;
715 GdipDeleteCustomLineCap(pen
->customend
);
716 GdipDeleteCustomLineCap(pen
->customstart
);
717 pen
->customend
= NULL
;
718 pen
->customstart
= NULL
;
720 pen
->startcap
= start
;
727 /* FIXME: Miter line joins behave a bit differently than they do in windows.
728 * Both kinds of miter joins clip if the angle is less than 11 degrees. */
729 GpStatus WINGDIPAPI
GdipSetPenLineJoin(GpPen
*pen
, GpLineJoin join
)
731 TRACE("(%p, %d)\n", pen
, join
);
733 if(!pen
) return InvalidParameter
;
736 pen
->style
&= ~(PS_JOIN_ROUND
| PS_JOIN_BEVEL
| PS_JOIN_MITER
);
737 pen
->style
|= gdip_to_gdi_join(join
);
742 GpStatus WINGDIPAPI
GdipSetPenMiterLimit(GpPen
*pen
, REAL limit
)
744 TRACE("(%p, %.2f)\n", pen
, limit
);
747 return InvalidParameter
;
749 pen
->miterlimit
= limit
;
754 GpStatus WINGDIPAPI
GdipSetPenStartCap(GpPen
*pen
, GpLineCap cap
)
756 TRACE("(%p, %d)\n", pen
, cap
);
758 if(!pen
) return InvalidParameter
;
760 GdipDeleteCustomLineCap(pen
->customstart
);
761 pen
->customstart
= NULL
;
767 GpStatus WINGDIPAPI
GdipSetPenWidth(GpPen
*pen
, REAL width
)
769 TRACE("(%p, %.2f)\n", pen
, width
);
771 if(!pen
) return InvalidParameter
;
778 GpStatus WINGDIPAPI
GdipSetPenMode(GpPen
*pen
, GpPenAlignment mode
)
780 TRACE("(%p, %d)\n", pen
, mode
);
782 if(!pen
) return InvalidParameter
;