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
29 #include "gdiplus_private.h"
30 #include "wine/debug.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(gdiplus
);
34 /* Multiplies two matrices of the form
40 * and puts the output in out.
42 static inline void matrix_multiply(GDIPCONST REAL
* left
, GDIPCONST REAL
* right
, REAL
* out
)
45 temp
[0] = left
[0] * right
[0] + left
[1] * right
[2];
46 temp
[1] = left
[0] * right
[1] + left
[1] * right
[3];
47 temp
[2] = left
[2] * right
[0] + left
[3] * right
[2];
48 temp
[3] = left
[2] * right
[1] + left
[3] * right
[3];
49 temp
[4] = left
[4] * right
[0] + left
[5] * right
[2] + right
[4];
50 temp
[5] = left
[4] * right
[1] + left
[5] * right
[3] + right
[5];
51 memcpy(out
, temp
, 6 * sizeof(REAL
));
54 static REAL
matrix_det(GDIPCONST GpMatrix
*matrix
)
56 return matrix
->matrix
[0]*matrix
->matrix
[3] - matrix
->matrix
[1]*matrix
->matrix
[2];
59 GpStatus WINGDIPAPI
GdipCreateMatrix2(REAL m11
, REAL m12
, REAL m21
, REAL m22
,
60 REAL dx
, REAL dy
, GpMatrix
**matrix
)
62 TRACE("(%.2f, %.2f, %.2f, %.2f, %.2f, %.2f, %p)\n", m11
, m12
, m21
, m22
, dx
, dy
, matrix
);
65 return InvalidParameter
;
67 *matrix
= heap_alloc_zero(sizeof(GpMatrix
));
68 if(!*matrix
) return OutOfMemory
;
71 (*matrix
)->matrix
[0] = m11
;
72 (*matrix
)->matrix
[1] = m12
;
74 (*matrix
)->matrix
[2] = m21
;
75 (*matrix
)->matrix
[3] = m22
;
77 (*matrix
)->matrix
[4] = dx
;
78 (*matrix
)->matrix
[5] = dy
;
83 GpStatus WINGDIPAPI
GdipCreateMatrix3(GDIPCONST GpRectF
*rect
,
84 GDIPCONST GpPointF
*pt
, GpMatrix
**matrix
)
86 REAL m11
, m12
, m21
, m22
, dx
, dy
;
88 TRACE("(%s, %p, %p)\n", debugstr_rectf(rect
), pt
, matrix
);
91 return InvalidParameter
;
93 m11
= (pt
[1].X
- pt
[0].X
) / rect
->Width
;
94 m21
= (pt
[2].X
- pt
[0].X
) / rect
->Height
;
95 dx
= pt
[0].X
- m11
* rect
->X
- m21
* rect
->Y
;
96 m12
= (pt
[1].Y
- pt
[0].Y
) / rect
->Width
;
97 m22
= (pt
[2].Y
- pt
[0].Y
) / rect
->Height
;
98 dy
= pt
[0].Y
- m12
* rect
->X
- m22
* rect
->Y
;
100 return GdipCreateMatrix2(m11
, m12
, m21
, m22
, dx
, dy
, matrix
);
103 GpStatus WINGDIPAPI
GdipCreateMatrix3I(GDIPCONST GpRect
*rect
, GDIPCONST GpPoint
*pt
,
110 TRACE("(%p, %p, %p)\n", rect
, pt
, matrix
);
112 set_rect(&rectF
, rect
->X
, rect
->Y
, rect
->Width
, rect
->Height
);
114 for (i
= 0; i
< 3; i
++) {
115 ptF
[i
].X
= (REAL
)pt
[i
].X
;
116 ptF
[i
].Y
= (REAL
)pt
[i
].Y
;
118 return GdipCreateMatrix3(&rectF
, ptF
, matrix
);
121 GpStatus WINGDIPAPI
GdipCloneMatrix(GpMatrix
*matrix
, GpMatrix
**clone
)
123 TRACE("(%s, %p)\n", debugstr_matrix(matrix
), clone
);
125 if(!matrix
|| !clone
)
126 return InvalidParameter
;
128 *clone
= heap_alloc_zero(sizeof(GpMatrix
));
129 if(!*clone
) return OutOfMemory
;
136 GpStatus WINGDIPAPI
GdipCreateMatrix(GpMatrix
**matrix
)
138 TRACE("(%p)\n", matrix
);
141 return InvalidParameter
;
143 *matrix
= heap_alloc_zero(sizeof(GpMatrix
));
144 if(!*matrix
) return OutOfMemory
;
146 (*matrix
)->matrix
[0] = 1.0;
147 (*matrix
)->matrix
[1] = 0.0;
148 (*matrix
)->matrix
[2] = 0.0;
149 (*matrix
)->matrix
[3] = 1.0;
150 (*matrix
)->matrix
[4] = 0.0;
151 (*matrix
)->matrix
[5] = 0.0;
156 GpStatus WINGDIPAPI
GdipDeleteMatrix(GpMatrix
*matrix
)
158 TRACE("(%p)\n", matrix
);
161 return InvalidParameter
;
168 GpStatus WINGDIPAPI
GdipGetMatrixElements(GDIPCONST GpMatrix
*matrix
,
171 TRACE("(%s, %p)\n", debugstr_matrix(matrix
), out
);
174 return InvalidParameter
;
176 memcpy(out
, matrix
->matrix
, sizeof(matrix
->matrix
));
181 GpStatus WINGDIPAPI
GdipInvertMatrix(GpMatrix
*matrix
)
186 TRACE("(%s)\n", debugstr_matrix(matrix
));
189 return InvalidParameter
;
191 /* optimize inverting simple scaling and translation matrices */
192 if(matrix
->matrix
[1] == 0 && matrix
->matrix
[2] == 0)
194 if (matrix
->matrix
[0] != 0 && matrix
->matrix
[3] != 0)
196 matrix
->matrix
[4] = -matrix
->matrix
[4] / matrix
->matrix
[0];
197 matrix
->matrix
[5] = -matrix
->matrix
[5] / matrix
->matrix
[3];
198 matrix
->matrix
[0] = 1 / matrix
->matrix
[0];
199 matrix
->matrix
[3] = 1 / matrix
->matrix
[3];
204 return InvalidParameter
;
206 det
= matrix_det(matrix
);
207 if (!(fabs(det
) >= 1e-5))
208 return InvalidParameter
;
214 matrix
->matrix
[0] = copy
.matrix
[3] * det
;
215 matrix
->matrix
[1] = -copy
.matrix
[1] * det
;
216 matrix
->matrix
[2] = -copy
.matrix
[2] * det
;
217 matrix
->matrix
[3] = copy
.matrix
[0] * det
;
218 matrix
->matrix
[4] = (copy
.matrix
[2]*copy
.matrix
[5]-copy
.matrix
[3]*copy
.matrix
[4]) * det
;
219 matrix
->matrix
[5] = -(copy
.matrix
[0]*copy
.matrix
[5]-copy
.matrix
[1]*copy
.matrix
[4]) * det
;
224 GpStatus WINGDIPAPI
GdipIsMatrixInvertible(GDIPCONST GpMatrix
*matrix
, BOOL
*result
)
226 TRACE("(%s, %p)\n", debugstr_matrix(matrix
), result
);
228 if(!matrix
|| !result
)
229 return InvalidParameter
;
231 if(matrix
->matrix
[1] == 0 && matrix
->matrix
[2] == 0)
232 *result
= matrix
->matrix
[0] != 0 && matrix
->matrix
[3] != 0;
234 *result
= (fabs(matrix_det(matrix
)) >= 1e-5);
239 GpStatus WINGDIPAPI
GdipMultiplyMatrix(GpMatrix
*matrix
, GDIPCONST GpMatrix
* matrix2
,
242 TRACE("(%s, %s, %d)\n", debugstr_matrix(matrix
), debugstr_matrix(matrix2
), order
);
244 if(!matrix
|| !matrix2
)
245 return InvalidParameter
;
247 if(order
== MatrixOrderAppend
)
248 matrix_multiply(matrix
->matrix
, matrix2
->matrix
, matrix
->matrix
);
249 else if (order
== MatrixOrderPrepend
)
250 matrix_multiply(matrix2
->matrix
, matrix
->matrix
, matrix
->matrix
);
252 return InvalidParameter
;
257 GpStatus WINGDIPAPI
GdipRotateMatrix(GpMatrix
*matrix
, REAL angle
,
260 REAL cos_theta
, sin_theta
, rotate
[6];
262 TRACE("(%p, %.2f, %d)\n", debugstr_matrix(matrix
), angle
, order
);
265 return InvalidParameter
;
267 angle
= deg2rad(angle
);
268 cos_theta
= cos(angle
);
269 sin_theta
= sin(angle
);
271 rotate
[0] = cos_theta
;
272 rotate
[1] = sin_theta
;
273 rotate
[2] = -sin_theta
;
274 rotate
[3] = cos_theta
;
278 if(order
== MatrixOrderAppend
)
279 matrix_multiply(matrix
->matrix
, rotate
, matrix
->matrix
);
280 else if (order
== MatrixOrderPrepend
)
281 matrix_multiply(rotate
, matrix
->matrix
, matrix
->matrix
);
283 return InvalidParameter
;
288 GpStatus WINGDIPAPI
GdipScaleMatrix(GpMatrix
*matrix
, REAL scaleX
, REAL scaleY
,
291 TRACE("(%s, %.2f, %.2f, %d)\n", debugstr_matrix(matrix
), scaleX
, scaleY
, order
);
294 return InvalidParameter
;
296 if(order
== MatrixOrderAppend
)
298 matrix
->matrix
[0] *= scaleX
;
299 matrix
->matrix
[1] *= scaleY
;
300 matrix
->matrix
[2] *= scaleX
;
301 matrix
->matrix
[3] *= scaleY
;
302 matrix
->matrix
[4] *= scaleX
;
303 matrix
->matrix
[5] *= scaleY
;
305 else if (order
== MatrixOrderPrepend
)
307 matrix
->matrix
[0] *= scaleX
;
308 matrix
->matrix
[1] *= scaleX
;
309 matrix
->matrix
[2] *= scaleY
;
310 matrix
->matrix
[3] *= scaleY
;
313 return InvalidParameter
;
318 GpStatus WINGDIPAPI
GdipSetMatrixElements(GpMatrix
*matrix
, REAL m11
, REAL m12
,
319 REAL m21
, REAL m22
, REAL dx
, REAL dy
)
321 TRACE("(%s, %.2f, %.2f, %.2f, %.2f, %.2f, %.2f)\n", debugstr_matrix(matrix
), m11
, m12
,
325 return InvalidParameter
;
327 matrix
->matrix
[0] = m11
;
328 matrix
->matrix
[1] = m12
;
329 matrix
->matrix
[2] = m21
;
330 matrix
->matrix
[3] = m22
;
331 matrix
->matrix
[4] = dx
;
332 matrix
->matrix
[5] = dy
;
337 GpStatus WINGDIPAPI
GdipShearMatrix(GpMatrix
*matrix
, REAL shearX
, REAL shearY
,
342 TRACE("(%s, %.2f, %.2f, %d)\n", debugstr_matrix(matrix
), shearX
, shearY
, order
);
345 return InvalidParameter
;
347 /* prepare transformation matrix */
355 if(order
== MatrixOrderAppend
)
356 matrix_multiply(matrix
->matrix
, shear
, matrix
->matrix
);
357 else if (order
== MatrixOrderPrepend
)
358 matrix_multiply(shear
, matrix
->matrix
, matrix
->matrix
);
360 return InvalidParameter
;
365 GpStatus WINGDIPAPI
GdipTransformMatrixPoints(GpMatrix
*matrix
, GpPointF
*pts
,
371 TRACE("(%s, %p, %d)\n", debugstr_matrix(matrix
), pts
, count
);
373 if(!matrix
|| !pts
|| count
<= 0)
374 return InvalidParameter
;
376 for(i
= 0; i
< count
; i
++)
381 pts
[i
].X
= x
* matrix
->matrix
[0] + y
* matrix
->matrix
[2] + matrix
->matrix
[4];
382 pts
[i
].Y
= x
* matrix
->matrix
[1] + y
* matrix
->matrix
[3] + matrix
->matrix
[5];
388 GpStatus WINGDIPAPI
GdipTransformMatrixPointsI(GpMatrix
*matrix
, GpPoint
*pts
, INT count
)
394 TRACE("(%s, %p, %d)\n", debugstr_matrix(matrix
), pts
, count
);
397 return InvalidParameter
;
399 ptsF
= heap_alloc_zero(sizeof(GpPointF
) * count
);
403 for(i
= 0; i
< count
; i
++){
404 ptsF
[i
].X
= (REAL
)pts
[i
].X
;
405 ptsF
[i
].Y
= (REAL
)pts
[i
].Y
;
408 ret
= GdipTransformMatrixPoints(matrix
, ptsF
, count
);
411 for(i
= 0; i
< count
; i
++){
412 pts
[i
].X
= gdip_round(ptsF
[i
].X
);
413 pts
[i
].Y
= gdip_round(ptsF
[i
].Y
);
420 GpStatus WINGDIPAPI
GdipTranslateMatrix(GpMatrix
*matrix
, REAL offsetX
,
421 REAL offsetY
, GpMatrixOrder order
)
423 TRACE("(%s, %.2f, %.2f, %d)\n", debugstr_matrix(matrix
), offsetX
, offsetY
, order
);
426 return InvalidParameter
;
428 if(order
== MatrixOrderAppend
)
430 matrix
->matrix
[4] += offsetX
;
431 matrix
->matrix
[5] += offsetY
;
433 else if (order
== MatrixOrderPrepend
)
435 matrix
->matrix
[4] = offsetX
* matrix
->matrix
[0] + offsetY
* matrix
->matrix
[2]
437 matrix
->matrix
[5] = offsetX
* matrix
->matrix
[1] + offsetY
* matrix
->matrix
[3]
441 return InvalidParameter
;
446 GpStatus WINGDIPAPI
GdipVectorTransformMatrixPoints(GpMatrix
*matrix
, GpPointF
*pts
, INT count
)
451 TRACE("(%s, %p, %d)\n", debugstr_matrix(matrix
), pts
, count
);
453 if(!matrix
|| !pts
|| count
<= 0)
454 return InvalidParameter
;
456 for(i
= 0; i
< count
; i
++)
461 pts
[i
].X
= x
* matrix
->matrix
[0] + y
* matrix
->matrix
[2];
462 pts
[i
].Y
= x
* matrix
->matrix
[1] + y
* matrix
->matrix
[3];
468 GpStatus WINGDIPAPI
GdipVectorTransformMatrixPointsI(GpMatrix
*matrix
, GpPoint
*pts
, INT count
)
474 TRACE("(%s, %p, %d)\n", debugstr_matrix(matrix
), pts
, count
);
477 return InvalidParameter
;
479 ptsF
= heap_alloc_zero(sizeof(GpPointF
) * count
);
483 for(i
= 0; i
< count
; i
++){
484 ptsF
[i
].X
= (REAL
)pts
[i
].X
;
485 ptsF
[i
].Y
= (REAL
)pts
[i
].Y
;
488 ret
= GdipVectorTransformMatrixPoints(matrix
, ptsF
, count
);
491 for(i
= 0; i
< count
; i
++){
492 pts
[i
].X
= gdip_round(ptsF
[i
].X
);
493 pts
[i
].Y
= gdip_round(ptsF
[i
].Y
);
500 GpStatus WINGDIPAPI
GdipIsMatrixEqual(GDIPCONST GpMatrix
*matrix
, GDIPCONST GpMatrix
*matrix2
,
503 TRACE("(%s, %s, %p)\n", debugstr_matrix(matrix
), debugstr_matrix(matrix2
), result
);
505 if(!matrix
|| !matrix2
|| !result
)
506 return InvalidParameter
;
507 /* based on single array member of GpMatrix */
508 *result
= (memcmp(matrix
->matrix
, matrix2
->matrix
, sizeof(GpMatrix
)) == 0);
513 GpStatus WINGDIPAPI
GdipIsMatrixIdentity(GDIPCONST GpMatrix
*matrix
, BOOL
*result
)
515 static const GpMatrix identity
=
522 TRACE("(%s, %p)\n", debugstr_matrix(matrix
), result
);
524 if(!matrix
|| !result
)
525 return InvalidParameter
;
527 return GdipIsMatrixEqual(matrix
, &identity
, result
);