Winspool DocumentProperties and DeviceCapabilities should now work on
[wine] / graphics / psdrv / graphics.c
1 /*
2  *      PostScript driver graphics functions
3  *
4  *      Copyright 1998  Huw D M Davies
5  *
6  */
7 #include <string.h>
8 #include <math.h>
9 #include "config.h"
10 #if defined(HAVE_FLOAT_H)
11  #include <float.h>
12 #endif
13 #if !defined(PI)
14  #define PI M_PI
15 #endif
16 #include "psdrv.h"
17 #include "debugtools.h"
18 #include "winspool.h"
19
20 DEFAULT_DEBUG_CHANNEL(psdrv)
21
22 /**********************************************************************
23  *           PSDRV_MoveToEx
24  */
25 BOOL PSDRV_MoveToEx(DC *dc, INT x, INT y, LPPOINT pt)
26 {
27     TRACE("%d %d\n", x, y);
28     if (pt)
29     {
30         pt->x = dc->w.CursPosX;
31         pt->y = dc->w.CursPosY;
32     }
33     dc->w.CursPosX = x;
34     dc->w.CursPosY = y;
35
36     return TRUE;
37 }
38
39
40 /***********************************************************************
41  *           PSDRV_LineTo
42  */
43 BOOL PSDRV_LineTo(DC *dc, INT x, INT y)
44 {
45     TRACE("%d %d\n", x, y);
46
47     PSDRV_SetPen(dc);
48     PSDRV_WriteMoveTo(dc, XLPTODP(dc, dc->w.CursPosX),
49                           YLPTODP(dc, dc->w.CursPosY));
50     PSDRV_WriteLineTo(dc, XLPTODP(dc, x), YLPTODP(dc, y));
51     PSDRV_WriteStroke(dc);
52
53     dc->w.CursPosX = x;
54     dc->w.CursPosY = y;
55     return TRUE;
56 }
57
58
59 /***********************************************************************
60  *           PSDRV_Rectangle
61  */
62 BOOL PSDRV_Rectangle( DC *dc, INT left, INT top, INT right,
63                        INT bottom )
64 {
65     INT width = XLSTODS(dc, right - left);
66     INT height = YLSTODS(dc, bottom - top);
67
68
69     TRACE("%d %d - %d %d\n", left, top, right, bottom);
70
71     PSDRV_WriteRectangle(dc, XLPTODP(dc, left), YLPTODP(dc, top),
72                              width, height);
73
74     PSDRV_Brush(dc,0);
75     PSDRV_SetPen(dc);
76     PSDRV_WriteStroke(dc);
77     return TRUE;
78 }
79
80
81 /***********************************************************************
82  *           PSDRV_RoundRect
83  */
84 BOOL PSDRV_RoundRect( DC *dc, INT left, INT top, INT right,
85                         INT bottom, INT ell_width, INT ell_height )
86 {
87     left = XLPTODP( dc, left );
88     right = XLPTODP( dc, right );
89     top = YLPTODP( dc, top );
90     bottom = YLPTODP( dc, bottom );
91     ell_width = XLSTODS( dc, ell_width );
92     ell_height = YLSTODS( dc, ell_height );
93
94     if( left > right ) { INT tmp = left; left = right; right = tmp; }
95     if( top > bottom ) { INT tmp = top; top = bottom; bottom = tmp; }
96
97     if(ell_width > right - left) ell_width = right - left;
98     if(ell_height > bottom - top) ell_height = bottom - top;
99
100     PSDRV_WriteMoveTo( dc, left, top + ell_height/2 );
101     PSDRV_WriteArc( dc, left + ell_width/2, top + ell_height/2, ell_width,
102                     ell_height, 90.0, 180.0);
103     PSDRV_WriteLineTo( dc, right - ell_width/2, top );
104     PSDRV_WriteArc( dc, right - ell_width/2, top + ell_height/2, ell_width,
105                     ell_height, 0.0, 90.0);
106     PSDRV_WriteLineTo( dc, right, bottom - ell_height/2 );
107     PSDRV_WriteArc( dc, right - ell_width/2, bottom - ell_height/2, ell_width,
108                     ell_height, -90.0, 0.0);
109     PSDRV_WriteLineTo( dc, right - ell_width/2, bottom);
110     PSDRV_WriteArc( dc, left + ell_width/2, bottom - ell_height/2, ell_width,
111                     ell_height, 180.0, -90.0);
112     PSDRV_WriteClosePath( dc );
113
114     PSDRV_Brush(dc,0);
115     PSDRV_SetPen(dc);
116     PSDRV_WriteStroke(dc);
117     return TRUE;
118 }
119
120 /***********************************************************************
121  *           PSDRV_DrawArc
122  *
123  * Does the work of Arc, Chord and Pie. lines is 0, 1 or 2 respectively.
124  */
125 static BOOL PSDRV_DrawArc( DC *dc, INT left, INT top, 
126                              INT right, INT bottom,
127                              INT xstart, INT ystart,
128                              INT xend, INT yend,
129                              int lines )
130 {
131     INT x, y, h, w;
132     double start_angle, end_angle, ratio;
133
134     x = XLPTODP(dc, (left + right)/2);
135     y = YLPTODP(dc, (top + bottom)/2);
136
137     w = XLSTODS(dc, (right - left));
138     h = YLSTODS(dc, (bottom - top));
139
140     if(w < 0) w = -w;
141     if(h < 0) h = -h;
142     ratio = ((double)w)/h;
143
144     /* angle is the angle after the rectangle is transformed to a square and is
145        measured anticlockwise from the +ve x-axis */
146
147     start_angle = atan2((double)(y - ystart) * ratio, (double)(xstart - x));
148     end_angle = atan2((double)(y - yend) * ratio, (double)(xend - x));
149
150     start_angle *= 180.0 / PI;
151     end_angle *= 180.0 / PI;
152
153     if(lines == 2) /* pie */
154         PSDRV_WriteMoveTo(dc, x, y);
155     else
156         PSDRV_WriteNewPath( dc );
157
158     PSDRV_WriteArc(dc, x, y, w, h, start_angle, end_angle);
159     if(lines == 1 || lines == 2) { /* chord or pie */
160         PSDRV_WriteClosePath(dc);
161         PSDRV_Brush(dc,0);
162     }
163     PSDRV_SetPen(dc);
164     PSDRV_WriteStroke(dc);
165     return TRUE;
166 }
167
168
169 /***********************************************************************
170  *           PSDRV_Arc
171  */
172 BOOL PSDRV_Arc( DC *dc, INT left, INT top, INT right, INT bottom,
173                   INT xstart, INT ystart, INT xend, INT yend )
174 {
175     return PSDRV_DrawArc( dc, left, top, right, bottom, xstart, ystart,
176                          xend, yend, 0 );
177 }
178
179 /***********************************************************************
180  *           PSDRV_Chord
181  */
182 BOOL PSDRV_Chord( DC *dc, INT left, INT top, INT right, INT bottom,
183                   INT xstart, INT ystart, INT xend, INT yend )
184 {
185     return PSDRV_DrawArc( dc, left, top, right, bottom, xstart, ystart,
186                          xend, yend, 1 );
187 }
188
189
190 /***********************************************************************
191  *           PSDRV_Pie
192  */
193 BOOL PSDRV_Pie( DC *dc, INT left, INT top, INT right, INT bottom,
194                   INT xstart, INT ystart, INT xend, INT yend )
195 {
196     return PSDRV_DrawArc( dc, left, top, right, bottom, xstart, ystart,
197                          xend, yend, 2 );
198 }
199
200
201 /***********************************************************************
202  *           PSDRV_Ellipse
203  */
204 BOOL PSDRV_Ellipse( DC *dc, INT left, INT top, INT right, INT bottom)
205 {
206     INT x, y, w, h;
207
208     TRACE("%d %d - %d %d\n", left, top, right, bottom);
209
210     x = XLPTODP(dc, (left + right)/2);
211     y = YLPTODP(dc, (top + bottom)/2);
212
213     w = XLSTODS(dc, (right - left));
214     h = YLSTODS(dc, (bottom - top));
215
216     PSDRV_WriteArc(dc, x, y, w, h, 0.0, 360.0);
217     PSDRV_WriteClosePath(dc);
218     PSDRV_Brush(dc,0);
219     PSDRV_SetPen(dc);
220     PSDRV_WriteStroke(dc);
221     return TRUE;
222 }
223
224
225 /***********************************************************************
226  *           PSDRV_PolyPolyline
227  */
228 BOOL PSDRV_PolyPolyline( DC *dc, const POINT* pts, const DWORD* counts,
229                            DWORD polylines )
230 {
231     DWORD polyline, line;
232     const POINT* pt;
233     TRACE("\n");
234
235     pt = pts;
236     for(polyline = 0; polyline < polylines; polyline++) {
237         PSDRV_WriteMoveTo(dc, XLPTODP(dc, pt->x), YLPTODP(dc, pt->y));
238         pt++;
239         for(line = 1; line < counts[polyline]; line++) {
240             PSDRV_WriteLineTo(dc, XLPTODP(dc, pt->x), YLPTODP(dc, pt->y));
241             pt++;
242         }
243     }
244     PSDRV_SetPen(dc);
245     PSDRV_WriteStroke(dc);
246     return TRUE;
247 }   
248
249
250 /***********************************************************************
251  *           PSDRV_Polyline
252  */
253 BOOL PSDRV_Polyline( DC *dc, const POINT* pt, INT count )
254 {
255     return PSDRV_PolyPolyline( dc, pt, (LPDWORD) &count, 1 );
256 }
257
258
259 /***********************************************************************
260  *           PSDRV_PolyPolygon
261  */
262 BOOL PSDRV_PolyPolygon( DC *dc, const POINT* pts, const INT* counts,
263                           UINT polygons )
264 {
265     DWORD polygon, line;
266     const POINT* pt;
267     TRACE("\n");
268
269     pt = pts;
270     for(polygon = 0; polygon < polygons; polygon++) {
271         PSDRV_WriteMoveTo(dc, XLPTODP(dc, pt->x), YLPTODP(dc, pt->y));
272         pt++;
273         for(line = 1; line < counts[polygon]; line++) {
274             PSDRV_WriteLineTo(dc, XLPTODP(dc, pt->x), YLPTODP(dc, pt->y));
275             pt++;
276         }
277         PSDRV_WriteClosePath(dc);
278     }
279
280     if(dc->w.polyFillMode == ALTERNATE)
281         PSDRV_Brush(dc, 1);
282     else /* WINDING */
283         PSDRV_Brush(dc, 0);
284     PSDRV_SetPen(dc);
285     PSDRV_WriteStroke(dc);
286     return TRUE;
287 }
288
289
290 /***********************************************************************
291  *           PSDRV_Polygon
292  */
293 BOOL PSDRV_Polygon( DC *dc, const POINT* pt, INT count )
294 {
295      return PSDRV_PolyPolygon( dc, pt, &count, 1 );
296 }
297
298
299 /***********************************************************************
300  *           PSDRV_SetPixel
301  */
302 COLORREF PSDRV_SetPixel( DC *dc, INT x, INT y, COLORREF color )
303 {
304     PSDRV_PDEVICE *physDev = (PSDRV_PDEVICE *)dc->physDev;
305     PSCOLOR pscolor;
306
307     x = XLPTODP(dc, x);
308     y = YLPTODP(dc, y);
309
310     PSDRV_WriteRectangle( dc, x, y, 0, 0 );
311     PSDRV_CreateColor( physDev, &pscolor, color );
312     PSDRV_WriteSetColor( dc, &pscolor );
313     PSDRV_WriteFill( dc );
314     return color;
315 }