4 * Copyright 1993 Alexandre Julliard
25 /* Object types for EnumObjects() */
29 /***********************************************************************
33 static BRUSHOBJ WhiteBrush
=
35 { 0, BRUSH_MAGIC
, 1 }, /* header */
36 { BS_SOLID
, RGB(255,255,255), 0 } /* logbrush */
39 static BRUSHOBJ LtGrayBrush
=
41 { 0, BRUSH_MAGIC
, 1 }, /* header */
42 { BS_SOLID
, RGB(192,192,192), 0 } /* logbrush */
45 static BRUSHOBJ GrayBrush
=
47 { 0, BRUSH_MAGIC
, 1 }, /* header */
48 { BS_SOLID
, RGB(128,128,128), 0 } /* logbrush */
51 static BRUSHOBJ DkGrayBrush
=
53 { 0, BRUSH_MAGIC
, 1 }, /* header */
54 { BS_SOLID
, RGB(64,64,64), 0 } /* logbrush */
57 static BRUSHOBJ BlackBrush
=
59 { 0, BRUSH_MAGIC
, 1 }, /* header */
60 { BS_SOLID
, RGB(0,0,0), 0 } /* logbrush */
63 static BRUSHOBJ NullBrush
=
65 { 0, BRUSH_MAGIC
, 1 }, /* header */
66 { BS_NULL
, 0, 0 } /* logbrush */
69 static PENOBJ WhitePen
=
71 { 0, PEN_MAGIC
, 1 }, /* header */
72 { PS_SOLID
, { 1, 0 }, RGB(255,255,255) } /* logpen */
75 static PENOBJ BlackPen
=
77 { 0, PEN_MAGIC
, 1 }, /* header */
78 { PS_SOLID
, { 1, 0 }, RGB(0,0,0) } /* logpen */
81 static PENOBJ NullPen
=
83 { 0, PEN_MAGIC
, 1 }, /* header */
84 { PS_NULL
, { 1, 0 }, 0 } /* logpen */
87 static FONTOBJ OEMFixedFont
=
89 { 0, FONT_MAGIC
, 1 }, /* header */
90 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, OEM_CHARSET
,
91 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
94 static FONTOBJ AnsiFixedFont
=
96 { 0, FONT_MAGIC
, 1 }, /* header */
97 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
98 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
101 static FONTOBJ AnsiVarFont
=
103 { 0, FONT_MAGIC
, 1 }, /* header */
104 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
105 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
108 static FONTOBJ SystemFont
=
110 { 0, FONT_MAGIC
, 1 }, /* header */
111 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
112 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
115 static FONTOBJ DeviceDefaultFont
=
117 { 0, FONT_MAGIC
, 1 }, /* header */
118 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
119 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
122 static FONTOBJ SystemFixedFont
=
124 { 0, FONT_MAGIC
, 1 }, /* header */
125 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
126 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
130 static GDIOBJHDR
* StockObjects
[NB_STOCK_OBJECTS
] =
132 (GDIOBJHDR
*) &WhiteBrush
,
133 (GDIOBJHDR
*) &LtGrayBrush
,
134 (GDIOBJHDR
*) &GrayBrush
,
135 (GDIOBJHDR
*) &DkGrayBrush
,
136 (GDIOBJHDR
*) &BlackBrush
,
137 (GDIOBJHDR
*) &NullBrush
,
138 (GDIOBJHDR
*) &WhitePen
,
139 (GDIOBJHDR
*) &BlackPen
,
140 (GDIOBJHDR
*) &NullPen
,
142 (GDIOBJHDR
*) &OEMFixedFont
,
143 (GDIOBJHDR
*) &AnsiFixedFont
,
144 (GDIOBJHDR
*) &AnsiVarFont
,
145 (GDIOBJHDR
*) &SystemFont
,
146 (GDIOBJHDR
*) &DeviceDefaultFont
,
147 NULL
, /* DEFAULT_PALETTE created by COLOR_Init */
148 (GDIOBJHDR
*) &SystemFixedFont
151 static FARPROC16 defDCHookCallback
;
154 /***********************************************************************
157 * GDI initialization.
159 BOOL32
GDI_Init(void)
162 extern BOOL32
X11DRV_Init(void);
164 /* Initialize drivers */
166 if (!X11DRV_Init()) return FALSE
;
168 /* Get default hook */
170 defDCHookCallback
= (FARPROC16
)MODULE_GetEntryPoint(GetModuleHandle("USER"),
172 dprintf_gdi( stddeb
, "DCHook: 16-bit callback is %08x\n",
173 (unsigned)defDCHookCallback
);
175 /* Create default palette */
177 if (!(hpalette
= COLOR_Init())) return FALSE
;
178 StockObjects
[DEFAULT_PALETTE
] = (GDIOBJHDR
*)GDI_HEAP_LIN_ADDR( hpalette
);
180 /* Create default bitmap */
182 if (!BITMAP_Init()) return FALSE
;
184 /* Initialize brush dithering */
186 if (!BRUSH_Init()) return FALSE
;
188 /* Initialize fonts */
190 if (!FONT_Init()) return FALSE
;
196 /***********************************************************************
199 FARPROC16
GDI_GetDefDCHook(void)
201 return defDCHookCallback
;
205 /***********************************************************************
208 HANDLE16
GDI_AllocObject( WORD size
, WORD magic
)
210 static DWORD count
= 0;
212 HANDLE handle
= GDI_HEAP_ALLOC( size
);
213 if (!handle
) return 0;
214 obj
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
217 obj
->dwCount
= ++count
;
222 /***********************************************************************
225 BOOL32
GDI_FreeObject( HANDLE16 handle
)
229 /* Can't free stock objects */
230 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
233 object
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
234 if (!object
) return FALSE
;
235 object
->wMagic
= 0; /* Mark it as invalid */
239 GDI_HEAP_FREE( handle
);
243 /***********************************************************************
246 * Return a pointer to the GDI object associated to the handle.
247 * Return NULL if the object has the wrong magic number.
249 GDIOBJHDR
* GDI_GetObjPtr( HANDLE16 handle
, WORD magic
)
251 GDIOBJHDR
* ptr
= NULL
;
253 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
254 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
256 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
257 if (!ptr
) return NULL
;
258 if ((magic
!= MAGIC_DONTCARE
) && (ptr
->wMagic
!= magic
)) return NULL
;
263 /***********************************************************************
264 * DeleteObject (GDI.69)
266 BOOL
DeleteObject( HGDIOBJ16 obj
)
268 /* Check if object is valid */
270 GDIOBJHDR
* header
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( obj
);
271 if (!header
) return FALSE
;
273 dprintf_gdi(stddeb
, "DeleteObject: %04x\n", obj
);
277 switch(header
->wMagic
)
279 case PEN_MAGIC
: return GDI_FreeObject( obj
);
280 case BRUSH_MAGIC
: return BRUSH_DeleteObject( obj
, (BRUSHOBJ
*)header
);
281 case FONT_MAGIC
: return GDI_FreeObject( obj
);
282 case PALETTE_MAGIC
: return PALETTE_DeleteObject(obj
,(PALETTEOBJ
*)header
);
283 case BITMAP_MAGIC
: return BITMAP_DeleteObject( obj
, (BITMAPOBJ
*)header
);
284 case REGION_MAGIC
: return REGION_DeleteObject( obj
, (RGNOBJ
*)header
);
290 /***********************************************************************
291 * GetStockObject (GDI.87)
293 HANDLE
GetStockObject( int obj
)
295 if ((obj
< 0) || (obj
>= NB_STOCK_OBJECTS
)) return 0;
296 if (!StockObjects
[obj
]) return 0;
297 dprintf_gdi(stddeb
, "GetStockObject: returning %d\n",
298 FIRST_STOCK_HANDLE
+ obj
);
299 return (HANDLE
)(FIRST_STOCK_HANDLE
+ obj
);
303 /***********************************************************************
304 * GetObject16 (GDI.82)
306 INT16
GetObject16( HANDLE16 handle
, INT16 count
, LPVOID buffer
)
308 GDIOBJHDR
* ptr
= NULL
;
309 dprintf_gdi(stddeb
, "GetObject16: %04x %d %p\n", handle
, count
, buffer
);
310 if (!count
) return 0;
312 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
313 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
315 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
321 return PEN_GetObject( (PENOBJ
*)ptr
, count
, buffer
);
323 return BRUSH_GetObject( (BRUSHOBJ
*)ptr
, count
, buffer
);
325 return BITMAP_GetObject16( (BITMAPOBJ
*)ptr
, count
, buffer
);
327 return FONT_GetObject16( (FONTOBJ
*)ptr
, count
, buffer
);
329 return PALETTE_GetObject( (PALETTEOBJ
*)ptr
, count
, buffer
);
335 /***********************************************************************
336 * GetObject32A (GDI32.204)
338 INT32
GetObject32A( HANDLE32 handle
, INT32 count
, LPVOID buffer
)
340 GDIOBJHDR
* ptr
= NULL
;
341 dprintf_gdi(stddeb
, "GetObject32A: %08x %d %p\n", handle
, count
, buffer
);
342 if (!count
) return 0;
344 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
345 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
347 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
353 return BITMAP_GetObject32( (BITMAPOBJ
*)ptr
, count
, buffer
);
355 return FONT_GetObject32A( (FONTOBJ
*)ptr
, count
, buffer
);
359 fprintf( stderr
, "GetObject32: magic %04x not implemented\n",
367 /***********************************************************************
368 * GetObject32W (GDI32.206)
370 INT32
GetObject32W( HANDLE32 handle
, INT32 count
, LPVOID buffer
)
372 return GetObject32A( handle
, count
, buffer
);
376 /***********************************************************************
377 * SelectObject (GDI.45)
379 HANDLE
SelectObject( HDC hdc
, HANDLE handle
)
381 GDIOBJHDR
* ptr
= NULL
;
384 dprintf_gdi(stddeb
, "SelectObject: hdc=%04x %04x\n", hdc
, handle
);
385 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
386 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
388 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
391 dc
= (DC
*) GDI_GetObjPtr( hdc
, DC_MAGIC
);
394 dc
= (DC
*)GDI_GetObjPtr(hdc
, METAFILE_DC_MAGIC
);
401 return PEN_SelectObject( dc
, handle
, (PENOBJ
*)ptr
);
403 return BRUSH_SelectObject( dc
, handle
, (BRUSHOBJ
*)ptr
);
405 return BITMAP_SelectObject( dc
, handle
, (BITMAPOBJ
*)ptr
);
407 return FONT_SelectObject( dc
, handle
, (FONTOBJ
*)ptr
);
409 return (HANDLE
)SelectClipRgn( hdc
, handle
);
415 /***********************************************************************
416 * UnrealizeObject (GDI.150)
418 BOOL
UnrealizeObject( HANDLE obj
)
420 /* Check if object is valid */
422 GDIOBJHDR
* header
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( obj
);
423 if (!header
) return FALSE
;
425 dprintf_gdi( stddeb
, "UnrealizeObject: %04x\n", obj
);
427 /* Unrealize object */
429 switch(header
->wMagic
)
432 return PALETTE_UnrealizeObject( obj
, (PALETTEOBJ
*)header
);
435 /* Windows resets the brush origin. We don't need to. */
442 /***********************************************************************
443 * EnumObjects (GDI.71)
445 INT
EnumObjects( HDC hdc
, INT nObjType
, GOBJENUMPROC16 lpEnumFunc
,
448 /* Solid colors to enumerate */
449 static const COLORREF solid_colors
[] =
450 { RGB(0x00,0x00,0x00), RGB(0xff,0xff,0xff),
451 RGB(0xff,0x00,0x00), RGB(0x00,0xff,0x00),
452 RGB(0x00,0x00,0xff), RGB(0xff,0xff,0x00),
453 RGB(0xff,0x00,0xff), RGB(0x00,0xff,0xff),
454 RGB(0x80,0x00,0x00), RGB(0x00,0x80,0x00),
455 RGB(0x80,0x80,0x00), RGB(0x00,0x00,0x80),
456 RGB(0x80,0x00,0x80), RGB(0x00,0x80,0x80),
457 RGB(0x80,0x80,0x80), RGB(0xc0,0xc0,0xc0)
462 LOGBRUSH16
*brush
= NULL
;
464 dprintf_gdi( stddeb
, "EnumObjects: %04x %d %08lx %08lx\n",
465 hdc
, nObjType
, (DWORD
)lpEnumFunc
, lParam
);
469 /* Enumerate solid pens */
470 if (!(pen
= SEGPTR_NEW(LOGPEN16
))) break;
471 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
473 pen
->lopnStyle
= PS_SOLID
;
474 pen
->lopnWidth
.x
= 1;
475 pen
->lopnWidth
.y
= 0;
476 pen
->lopnColor
= solid_colors
[i
];
477 retval
= lpEnumFunc( SEGPTR_GET(pen
), lParam
);
478 dprintf_gdi( stddeb
, "EnumObject: solid pen %08lx, ret=%d\n",
479 solid_colors
[i
], retval
);
486 /* Enumerate solid brushes */
487 if (!(brush
= SEGPTR_NEW(LOGBRUSH16
))) break;
488 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
490 brush
->lbStyle
= BS_SOLID
;
491 brush
->lbColor
= solid_colors
[i
];
493 retval
= lpEnumFunc( SEGPTR_GET(brush
), lParam
);
494 dprintf_gdi( stddeb
, "EnumObject: solid brush %08lx, ret=%d\n",
495 solid_colors
[i
], retval
);
499 /* Now enumerate hatched brushes */
500 if (retval
) for (i
= HS_HORIZONTAL
; i
<= HS_DIAGCROSS
; i
++)
502 brush
->lbStyle
= BS_HATCHED
;
503 brush
->lbColor
= RGB(0,0,0);
505 retval
= lpEnumFunc( SEGPTR_GET(brush
), lParam
);
506 dprintf_gdi( stddeb
, "EnumObject: hatched brush %d, ret=%d\n",
514 fprintf( stderr
, "EnumObjects: invalid type %d\n", nObjType
);
521 /***********************************************************************
522 * IsGDIObject (GDI.462)
524 BOOL
IsGDIObject(HANDLE handle
)
526 GDIOBJHDR
*object
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
528 return (object
->wMagic
>=PEN_MAGIC
&& object
->wMagic
<= METAFILE_DC_MAGIC
);
533 /***********************************************************************
536 INT16
MulDiv16( INT16 foo
, INT16 bar
, INT16 baz
)
539 if (!baz
) return -32768;
540 ret
= (foo
* bar
) / baz
;
541 if ((ret
> 32767) || (ret
< -32767)) return -32768;
546 /***********************************************************************
547 * MulDiv32 (KERNEL32.391)
549 INT32
MulDiv32( INT32 foo
, INT32 bar
, INT32 baz
)
554 ret
= ((long long)foo
* bar
) / baz
;
555 if ((ret
> 2147483647) || (ret
< -2147483647)) return -1;
559 return (foo
* bar
) / baz
;