4 * Copyright 1993 Alexandre Julliard
23 /* Object types for EnumObjects() */
27 /***********************************************************************
31 static BRUSHOBJ WhiteBrush
=
33 { 0, BRUSH_MAGIC
, 1 }, /* header */
34 { BS_SOLID
, RGB(255,255,255), 0 } /* logbrush */
37 static BRUSHOBJ LtGrayBrush
=
39 { 0, BRUSH_MAGIC
, 1 }, /* header */
40 /* FIXME : this should perhaps be BS_HATCHED, at least for 1 bitperpixel */
41 { BS_SOLID
, RGB(192,192,192), 0 } /* logbrush */
44 static BRUSHOBJ GrayBrush
=
46 { 0, BRUSH_MAGIC
, 1 }, /* header */
47 /* FIXME : this should perhaps be BS_HATCHED, at least for 1 bitperpixel */
48 { BS_SOLID
, RGB(128,128,128), 0 } /* logbrush */
51 static BRUSHOBJ DkGrayBrush
=
53 { 0, BRUSH_MAGIC
, 1 }, /* header */
54 /* This is BS_HATCHED, for 1 bitperpixel. This makes the spray work in pbrush */
55 /* NB_HATCH_STYLES is an index into HatchBrushes */
56 { BS_HATCHED
, RGB(0,0,0), NB_HATCH_STYLES
} /* logbrush */
59 static BRUSHOBJ BlackBrush
=
61 { 0, BRUSH_MAGIC
, 1 }, /* header */
62 { BS_SOLID
, RGB(0,0,0), 0 } /* logbrush */
65 static BRUSHOBJ NullBrush
=
67 { 0, BRUSH_MAGIC
, 1 }, /* header */
68 { BS_NULL
, 0, 0 } /* logbrush */
71 static PENOBJ WhitePen
=
73 { 0, PEN_MAGIC
, 1 }, /* header */
74 { PS_SOLID
, { 1, 0 }, RGB(255,255,255) } /* logpen */
77 static PENOBJ BlackPen
=
79 { 0, PEN_MAGIC
, 1 }, /* header */
80 { PS_SOLID
, { 1, 0 }, RGB(0,0,0) } /* logpen */
83 static PENOBJ NullPen
=
85 { 0, PEN_MAGIC
, 1 }, /* header */
86 { PS_NULL
, { 1, 0 }, 0 } /* logpen */
89 static FONTOBJ OEMFixedFont
=
91 { 0, FONT_MAGIC
, 1 }, /* header */
92 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, OEM_CHARSET
,
93 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
96 static FONTOBJ AnsiFixedFont
=
98 { 0, FONT_MAGIC
, 1 }, /* header */
99 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
100 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
103 static FONTOBJ AnsiVarFont
=
105 { 0, FONT_MAGIC
, 1 }, /* header */
106 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
107 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "MS Sans Serif" }
110 static FONTOBJ SystemFont
=
112 { 0, FONT_MAGIC
, 1 },
113 { 16, 0, 0, 0, FW_BOLD
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
114 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "System" }
117 static FONTOBJ DeviceDefaultFont
=
119 { 0, FONT_MAGIC
, 1 }, /* header */
120 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
121 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
124 static FONTOBJ SystemFixedFont
=
126 { 0, FONT_MAGIC
, 1 }, /* header */
127 { 12, 0, 0, 0, FW_BOLD
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
128 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
132 static GDIOBJHDR
* StockObjects
[NB_STOCK_OBJECTS
] =
134 (GDIOBJHDR
*) &WhiteBrush
,
135 (GDIOBJHDR
*) &LtGrayBrush
,
136 (GDIOBJHDR
*) &GrayBrush
,
137 (GDIOBJHDR
*) &DkGrayBrush
,
138 (GDIOBJHDR
*) &BlackBrush
,
139 (GDIOBJHDR
*) &NullBrush
,
140 (GDIOBJHDR
*) &WhitePen
,
141 (GDIOBJHDR
*) &BlackPen
,
142 (GDIOBJHDR
*) &NullPen
,
144 (GDIOBJHDR
*) &OEMFixedFont
,
145 (GDIOBJHDR
*) &AnsiFixedFont
,
146 (GDIOBJHDR
*) &AnsiVarFont
,
147 (GDIOBJHDR
*) &SystemFont
,
148 (GDIOBJHDR
*) &DeviceDefaultFont
,
149 NULL
, /* DEFAULT_PALETTE created by PALETTE_Init */
150 (GDIOBJHDR
*) &SystemFixedFont
154 /***********************************************************************
157 * GDI initialization.
159 BOOL32
GDI_Init(void)
161 extern BOOL32
X11DRV_Init(void);
162 extern BOOL32
DIB_Init(void);
164 /* Initialize drivers */
166 DIB_Init(); /* always before X11DRV_Init() */
170 /* Create default palette */
172 HPALETTE16 hpalette
= PALETTE_Init();
176 StockObjects
[DEFAULT_PALETTE
] = (GDIOBJHDR
*)GDI_HEAP_LIN_ADDR( hpalette
);
184 /***********************************************************************
187 HGDIOBJ16
GDI_AllocObject( WORD size
, WORD magic
)
189 static DWORD count
= 0;
191 HGDIOBJ16 handle
= GDI_HEAP_ALLOC( size
);
192 if (!handle
) return 0;
193 obj
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
196 obj
->dwCount
= ++count
;
201 /***********************************************************************
204 BOOL32
GDI_FreeObject( HGDIOBJ16 handle
)
208 /* Can't free stock objects */
209 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
212 object
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
213 if (!object
) return FALSE
;
214 object
->wMagic
= 0; /* Mark it as invalid */
218 GDI_HEAP_FREE( handle
);
222 /***********************************************************************
225 * Return a pointer to the GDI object associated to the handle.
226 * Return NULL if the object has the wrong magic number.
228 GDIOBJHDR
* GDI_GetObjPtr( HGDIOBJ16 handle
, WORD magic
)
230 GDIOBJHDR
* ptr
= NULL
;
232 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
233 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
235 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
236 if (!ptr
) return NULL
;
237 if ((magic
!= MAGIC_DONTCARE
) && (ptr
->wMagic
!= magic
)) return NULL
;
242 /***********************************************************************
243 * DeleteObject16 (GDI.69)
245 BOOL16
DeleteObject16( HGDIOBJ16 obj
)
247 return DeleteObject32( obj
);
251 /***********************************************************************
252 * DeleteObject32 (GDI32.70)
254 BOOL32
DeleteObject32( HGDIOBJ32 obj
)
256 /* Check if object is valid */
258 GDIOBJHDR
* header
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( obj
);
259 if (!header
|| HIWORD(obj
)) return FALSE
;
261 dprintf_gdi(stddeb
, "DeleteObject: %04x\n", obj
);
265 switch(header
->wMagic
)
267 case PEN_MAGIC
: return GDI_FreeObject( obj
);
268 case BRUSH_MAGIC
: return BRUSH_DeleteObject( obj
, (BRUSHOBJ
*)header
);
269 case FONT_MAGIC
: return GDI_FreeObject( obj
);
270 case PALETTE_MAGIC
: return PALETTE_DeleteObject(obj
,(PALETTEOBJ
*)header
);
271 case BITMAP_MAGIC
: return BITMAP_DeleteObject( obj
, (BITMAPOBJ
*)header
);
272 case REGION_MAGIC
: return REGION_DeleteObject( obj
, (RGNOBJ
*)header
);
278 /***********************************************************************
279 * GetStockObject16 (GDI.87)
281 HGDIOBJ16
GetStockObject16( INT16 obj
)
283 return (HGDIOBJ16
)GetStockObject32( obj
);
287 /***********************************************************************
288 * GetStockObject32 (GDI32.220)
290 HGDIOBJ32
GetStockObject32( INT32 obj
)
292 if ((obj
< 0) || (obj
>= NB_STOCK_OBJECTS
)) return 0;
293 if (!StockObjects
[obj
]) return 0;
294 dprintf_gdi(stddeb
, "GetStockObject: returning %d\n",
295 FIRST_STOCK_HANDLE
+ obj
);
296 return (HGDIOBJ16
)(FIRST_STOCK_HANDLE
+ obj
);
300 /***********************************************************************
301 * GetObject16 (GDI.82)
303 INT16
GetObject16( HANDLE16 handle
, INT16 count
, LPVOID buffer
)
305 GDIOBJHDR
* ptr
= NULL
;
306 dprintf_gdi(stddeb
, "GetObject16: %04x %d %p\n", handle
, count
, buffer
);
307 if (!count
) return 0;
309 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
310 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
312 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
318 return PEN_GetObject16( (PENOBJ
*)ptr
, count
, buffer
);
320 return BRUSH_GetObject16( (BRUSHOBJ
*)ptr
, count
, buffer
);
322 return BITMAP_GetObject16( (BITMAPOBJ
*)ptr
, count
, buffer
);
324 return FONT_GetObject16( (FONTOBJ
*)ptr
, count
, buffer
);
326 return PALETTE_GetObject( (PALETTEOBJ
*)ptr
, count
, buffer
);
332 /***********************************************************************
333 * GetObject32A (GDI32.204)
335 INT32
GetObject32A( HANDLE32 handle
, INT32 count
, LPVOID buffer
)
337 GDIOBJHDR
* ptr
= NULL
;
338 dprintf_gdi(stddeb
, "GetObject32A: %08x %d %p\n", handle
, count
, buffer
);
339 if (!count
) return 0;
341 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
342 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
344 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
350 return PEN_GetObject32( (PENOBJ
*)ptr
, count
, buffer
);
352 return BRUSH_GetObject32( (BRUSHOBJ
*)ptr
, count
, buffer
);
354 return BITMAP_GetObject32( (BITMAPOBJ
*)ptr
, count
, buffer
);
356 return FONT_GetObject32A( (FONTOBJ
*)ptr
, count
, buffer
);
358 fprintf( stderr
, "GetObject32: magic %04x not implemented\n",
366 /***********************************************************************
367 * GetObject32W (GDI32.206)
369 INT32
GetObject32W( HANDLE32 handle
, INT32 count
, LPVOID buffer
)
371 return GetObject32A( handle
, count
, buffer
);
375 /***********************************************************************
376 * SelectObject16 (GDI.45)
378 HGDIOBJ16
SelectObject16( HDC16 hdc
, HGDIOBJ16 handle
)
380 return (HGDIOBJ16
)SelectObject32( hdc
, handle
);
384 /***********************************************************************
385 * SelectObject32 (GDI32.299)
387 HGDIOBJ32
SelectObject32( HDC32 hdc
, HGDIOBJ32 handle
)
389 DC
* dc
= DC_GetDCPtr( hdc
);
390 if (!dc
|| !dc
->funcs
->pSelectObject
) return 0;
391 dprintf_gdi(stddeb
, "SelectObject: hdc=%04x %04x\n", hdc
, handle
);
392 return dc
->funcs
->pSelectObject( dc
, handle
);
396 /***********************************************************************
397 * UnrealizeObject16 (GDI.150)
399 BOOL16
UnrealizeObject16( HGDIOBJ16 obj
)
401 return UnrealizeObject32( obj
);
405 /***********************************************************************
406 * UnrealizeObject (GDI32.358)
408 BOOL32
UnrealizeObject32( HGDIOBJ32 obj
)
410 /* Check if object is valid */
412 GDIOBJHDR
* header
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( obj
);
413 if (!header
) return FALSE
;
415 dprintf_gdi( stddeb
, "UnrealizeObject: %04x\n", obj
);
417 /* Unrealize object */
419 switch(header
->wMagic
)
422 return PALETTE_UnrealizeObject( obj
, (PALETTEOBJ
*)header
);
425 /* Windows resets the brush origin. We don't need to. */
432 /***********************************************************************
433 * EnumObjects16 (GDI.71)
435 INT16
EnumObjects16( HDC16 hdc
, INT16 nObjType
, GOBJENUMPROC16 lpEnumFunc
,
438 /* Solid colors to enumerate */
439 static const COLORREF solid_colors
[] =
440 { RGB(0x00,0x00,0x00), RGB(0xff,0xff,0xff),
441 RGB(0xff,0x00,0x00), RGB(0x00,0xff,0x00),
442 RGB(0x00,0x00,0xff), RGB(0xff,0xff,0x00),
443 RGB(0xff,0x00,0xff), RGB(0x00,0xff,0xff),
444 RGB(0x80,0x00,0x00), RGB(0x00,0x80,0x00),
445 RGB(0x80,0x80,0x00), RGB(0x00,0x00,0x80),
446 RGB(0x80,0x00,0x80), RGB(0x00,0x80,0x80),
447 RGB(0x80,0x80,0x80), RGB(0xc0,0xc0,0xc0)
452 LOGBRUSH16
*brush
= NULL
;
454 dprintf_gdi( stddeb
, "EnumObjects16: %04x %d %08lx %08lx\n",
455 hdc
, nObjType
, (DWORD
)lpEnumFunc
, lParam
);
459 /* Enumerate solid pens */
460 if (!(pen
= SEGPTR_NEW(LOGPEN16
))) break;
461 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
463 pen
->lopnStyle
= PS_SOLID
;
464 pen
->lopnWidth
.x
= 1;
465 pen
->lopnWidth
.y
= 0;
466 pen
->lopnColor
= solid_colors
[i
];
467 retval
= lpEnumFunc( SEGPTR_GET(pen
), lParam
);
468 dprintf_gdi( stddeb
, "EnumObjects16: solid pen %08lx, ret=%d\n",
469 solid_colors
[i
], retval
);
476 /* Enumerate solid brushes */
477 if (!(brush
= SEGPTR_NEW(LOGBRUSH16
))) break;
478 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
480 brush
->lbStyle
= BS_SOLID
;
481 brush
->lbColor
= solid_colors
[i
];
483 retval
= lpEnumFunc( SEGPTR_GET(brush
), lParam
);
484 dprintf_gdi( stddeb
, "EnumObjects16: solid brush %08lx, ret=%d\n",
485 solid_colors
[i
], retval
);
489 /* Now enumerate hatched brushes */
490 if (retval
) for (i
= HS_HORIZONTAL
; i
<= HS_DIAGCROSS
; i
++)
492 brush
->lbStyle
= BS_HATCHED
;
493 brush
->lbColor
= RGB(0,0,0);
495 retval
= lpEnumFunc( SEGPTR_GET(brush
), lParam
);
496 dprintf_gdi( stddeb
, "EnumObjects16: hatched brush %d, ret=%d\n",
504 fprintf( stderr
, "EnumObjects16: invalid type %d\n", nObjType
);
511 /***********************************************************************
512 * EnumObjects32 (GDI32.89)
514 INT32
EnumObjects32( HDC32 hdc
, INT32 nObjType
, GOBJENUMPROC32 lpEnumFunc
,
517 /* Solid colors to enumerate */
518 static const COLORREF solid_colors
[] =
519 { RGB(0x00,0x00,0x00), RGB(0xff,0xff,0xff),
520 RGB(0xff,0x00,0x00), RGB(0x00,0xff,0x00),
521 RGB(0x00,0x00,0xff), RGB(0xff,0xff,0x00),
522 RGB(0xff,0x00,0xff), RGB(0x00,0xff,0xff),
523 RGB(0x80,0x00,0x00), RGB(0x00,0x80,0x00),
524 RGB(0x80,0x80,0x00), RGB(0x00,0x00,0x80),
525 RGB(0x80,0x00,0x80), RGB(0x00,0x80,0x80),
526 RGB(0x80,0x80,0x80), RGB(0xc0,0xc0,0xc0)
533 dprintf_gdi( stddeb
, "EnumObjects32: %04x %d %08lx %08lx\n",
534 hdc
, nObjType
, (DWORD
)lpEnumFunc
, lParam
);
538 /* Enumerate solid pens */
539 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
541 pen
.lopnStyle
= PS_SOLID
;
544 pen
.lopnColor
= solid_colors
[i
];
545 retval
= lpEnumFunc( &pen
, lParam
);
546 dprintf_gdi( stddeb
, "EnumObjects32: solid pen %08lx, ret=%d\n",
547 solid_colors
[i
], retval
);
553 /* Enumerate solid brushes */
554 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
556 brush
.lbStyle
= BS_SOLID
;
557 brush
.lbColor
= solid_colors
[i
];
559 retval
= lpEnumFunc( &brush
, lParam
);
560 dprintf_gdi( stddeb
, "EnumObjects32: solid brush %08lx, ret=%d\n",
561 solid_colors
[i
], retval
);
565 /* Now enumerate hatched brushes */
566 if (retval
) for (i
= HS_HORIZONTAL
; i
<= HS_DIAGCROSS
; i
++)
568 brush
.lbStyle
= BS_HATCHED
;
569 brush
.lbColor
= RGB(0,0,0);
571 retval
= lpEnumFunc( &brush
, lParam
);
572 dprintf_gdi( stddeb
, "EnumObjects32: hatched brush %d, ret=%d\n",
579 /* FIXME: implement Win32 types */
580 fprintf( stderr
, "EnumObjects32: invalid type %d\n", nObjType
);
587 /***********************************************************************
588 * IsGDIObject (GDI.462)
590 BOOL16
IsGDIObject( HGDIOBJ16 handle
)
592 GDIOBJHDR
*object
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
594 return (object
->wMagic
>=PEN_MAGIC
&& object
->wMagic
<= METAFILE_DC_MAGIC
);
599 /***********************************************************************
602 INT16
MulDiv16( INT16 foo
, INT16 bar
, INT16 baz
)
605 if (!baz
) return -32768;
606 ret
= (foo
* bar
) / baz
;
607 if ((ret
> 32767) || (ret
< -32767)) return -32768;
612 /***********************************************************************
613 * MulDiv32 (KERNEL32.391)
615 INT32
MulDiv32( INT32 foo
, INT32 bar
, INT32 baz
)
620 ret
= ((long long)foo
* bar
) / baz
;
621 if ((ret
> 2147483647) || (ret
< -2147483647)) return -1;
625 return (foo
* bar
) / baz
;