4 * Copyright 1993 Alexandre Julliard
26 /* Object types for EnumObjects() */
30 /***********************************************************************
34 static BRUSHOBJ WhiteBrush
=
36 { 0, BRUSH_MAGIC
, 1, 0 }, /* header */
37 { BS_SOLID
, RGB(255,255,255), 0 } /* logbrush */
40 static BRUSHOBJ LtGrayBrush
=
42 { 0, BRUSH_MAGIC
, 1, 0 }, /* header */
43 { BS_SOLID
, RGB(192,192,192), 0 } /* logbrush */
46 static BRUSHOBJ GrayBrush
=
48 { 0, BRUSH_MAGIC
, 1, 0 }, /* header */
49 { BS_SOLID
, RGB(128,128,128), 0 } /* logbrush */
52 static BRUSHOBJ DkGrayBrush
=
54 { 0, BRUSH_MAGIC
, 1, 0 }, /* header */
55 { BS_SOLID
, RGB(64,64,64), 0 } /* logbrush */
58 static BRUSHOBJ BlackBrush
=
60 { 0, BRUSH_MAGIC
, 1, 0 }, /* header */
61 { BS_SOLID
, RGB(0,0,0), 0 } /* logbrush */
64 static BRUSHOBJ NullBrush
=
66 { 0, BRUSH_MAGIC
, 1, 0 }, /* header */
67 { BS_NULL
, 0, 0 } /* logbrush */
70 static PENOBJ WhitePen
=
72 { 0, PEN_MAGIC
, 1, 0 }, /* header */
73 { PS_SOLID
, { 1, 0 }, RGB(255,255,255) } /* logpen */
76 static PENOBJ BlackPen
=
78 { 0, PEN_MAGIC
, 1, 0 }, /* header */
79 { PS_SOLID
, { 1, 0 }, RGB(0,0,0) } /* logpen */
82 static PENOBJ NullPen
=
84 { 0, PEN_MAGIC
, 1, 0 }, /* header */
85 { PS_NULL
, { 1, 0 }, 0 } /* logpen */
88 static FONTOBJ OEMFixedFont
=
90 { 0, FONT_MAGIC
, 1, 0 }, /* header */
91 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, OEM_CHARSET
,
92 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
95 static FONTOBJ AnsiFixedFont
=
97 { 0, FONT_MAGIC
, 1, 0 }, /* header */
98 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
99 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
102 static FONTOBJ AnsiVarFont
=
104 { 0, FONT_MAGIC
, 1, 0 }, /* header */
105 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
106 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
109 static FONTOBJ SystemFont
=
111 { 0, FONT_MAGIC
, 1, 0 }, /* header */
112 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
113 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
116 static FONTOBJ DeviceDefaultFont
=
118 { 0, FONT_MAGIC
, 1, 0 }, /* header */
119 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
120 0, 0, DEFAULT_QUALITY
, VARIABLE_PITCH
| FF_SWISS
, "" }
123 static FONTOBJ SystemFixedFont
=
125 { 0, FONT_MAGIC
, 1, 0 }, /* header */
126 { 12, 0, 0, 0, FW_NORMAL
, FALSE
, FALSE
, FALSE
, ANSI_CHARSET
,
127 0, 0, DEFAULT_QUALITY
, FIXED_PITCH
| FF_MODERN
, "" }
131 static GDIOBJHDR
* StockObjects
[NB_STOCK_OBJECTS
] =
133 (GDIOBJHDR
*) &WhiteBrush
,
134 (GDIOBJHDR
*) &LtGrayBrush
,
135 (GDIOBJHDR
*) &GrayBrush
,
136 (GDIOBJHDR
*) &DkGrayBrush
,
137 (GDIOBJHDR
*) &BlackBrush
,
138 (GDIOBJHDR
*) &NullBrush
,
139 (GDIOBJHDR
*) &WhitePen
,
140 (GDIOBJHDR
*) &BlackPen
,
141 (GDIOBJHDR
*) &NullPen
,
143 (GDIOBJHDR
*) &OEMFixedFont
,
144 (GDIOBJHDR
*) &AnsiFixedFont
,
145 (GDIOBJHDR
*) &AnsiVarFont
,
146 (GDIOBJHDR
*) &SystemFont
,
147 (GDIOBJHDR
*) &DeviceDefaultFont
,
148 NULL
, /* DEFAULT_PALETTE created by COLOR_Init */
149 (GDIOBJHDR
*) &SystemFixedFont
152 static FARPROC16 defDCHookCallback
;
155 /***********************************************************************
158 * GDI initialization.
160 BOOL32
GDI_Init(void)
164 defDCHookCallback
= (FARPROC16
)MODULE_GetEntryPoint(GetModuleHandle("USER"),
166 dprintf_gdi( stddeb
, "DCHook: 16-bit callback is %08x\n",
167 (unsigned)defDCHookCallback
);
169 /* Create default palette */
171 if (!(hpalette
= COLOR_Init())) return FALSE
;
172 StockObjects
[DEFAULT_PALETTE
] = (GDIOBJHDR
*)GDI_HEAP_LIN_ADDR( hpalette
);
174 /* Create default bitmap */
176 if (!BITMAP_Init()) return FALSE
;
178 /* Initialize brush dithering */
180 if (!BRUSH_Init()) return FALSE
;
182 /* Initialize fonts */
184 if (!FONT_Init()) return FALSE
;
190 /***********************************************************************
193 FARPROC16
GDI_GetDefDCHook(void)
195 return defDCHookCallback
;
199 /***********************************************************************
202 HANDLE16
GDI_AllocObject( WORD size
, WORD magic
)
204 static DWORD count
= 0;
206 HANDLE handle
= GDI_HEAP_ALLOC( size
);
207 if (!handle
) return 0;
208 obj
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
211 obj
->dwCount
= ++count
;
216 /***********************************************************************
219 BOOL32
GDI_FreeObject( HANDLE16 handle
)
223 /* Can't free stock objects */
224 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
227 object
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
228 if (!object
) return FALSE
;
229 object
->wMagic
= 0; /* Mark it as invalid */
233 GDI_HEAP_FREE( handle
);
237 /***********************************************************************
240 * Return a pointer to the GDI object associated to the handle.
241 * Return NULL if the object has the wrong magic number.
243 GDIOBJHDR
* GDI_GetObjPtr( HANDLE16 handle
, WORD magic
)
245 GDIOBJHDR
* ptr
= NULL
;
247 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
248 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
250 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
251 if (!ptr
) return NULL
;
252 if ((magic
!= MAGIC_DONTCARE
) && (ptr
->wMagic
!= magic
)) return NULL
;
257 /***********************************************************************
258 * DeleteObject (GDI.69)
260 BOOL
DeleteObject( HGDIOBJ16 obj
)
262 /* Check if object is valid */
264 GDIOBJHDR
* header
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( obj
);
265 if (!header
) return FALSE
;
267 dprintf_gdi(stddeb
, "DeleteObject: %04x\n", obj
);
271 switch(header
->wMagic
)
273 case PEN_MAGIC
: return GDI_FreeObject( obj
);
274 case BRUSH_MAGIC
: return BRUSH_DeleteObject( obj
, (BRUSHOBJ
*)header
);
275 case FONT_MAGIC
: return GDI_FreeObject( obj
);
276 case PALETTE_MAGIC
: return PALETTE_DeleteObject(obj
,(PALETTEOBJ
*)header
);
277 case BITMAP_MAGIC
: return BITMAP_DeleteObject( obj
, (BITMAPOBJ
*)header
);
278 case REGION_MAGIC
: return REGION_DeleteObject( obj
, (RGNOBJ
*)header
);
284 /***********************************************************************
285 * GetStockObject (GDI.87)
287 HANDLE
GetStockObject( int obj
)
289 if ((obj
< 0) || (obj
>= NB_STOCK_OBJECTS
)) return 0;
290 if (!StockObjects
[obj
]) return 0;
291 dprintf_gdi(stddeb
, "GetStockObject: returning %d\n",
292 FIRST_STOCK_HANDLE
+ obj
);
293 return (HANDLE
)(FIRST_STOCK_HANDLE
+ obj
);
297 /***********************************************************************
298 * GetObject16 (GDI.82)
300 INT16
GetObject16( HANDLE16 handle
, INT16 count
, LPVOID buffer
)
302 GDIOBJHDR
* ptr
= NULL
;
303 dprintf_gdi(stddeb
, "GetObject16: %04x %d %p\n", handle
, count
, buffer
);
304 if (!count
) return 0;
306 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
307 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
309 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
315 return PEN_GetObject( (PENOBJ
*)ptr
, count
, buffer
);
317 return BRUSH_GetObject( (BRUSHOBJ
*)ptr
, count
, buffer
);
319 return BITMAP_GetObject16( (BITMAPOBJ
*)ptr
, count
, buffer
);
321 return FONT_GetObject( (FONTOBJ
*)ptr
, count
, buffer
);
323 return PALETTE_GetObject( (PALETTEOBJ
*)ptr
, count
, buffer
);
329 /***********************************************************************
330 * GetObject32A (GDI32.204)
332 INT32
GetObject32A( HANDLE32 handle
, INT32 count
, LPVOID buffer
)
334 GDIOBJHDR
* ptr
= NULL
;
335 dprintf_gdi(stddeb
, "GetObject32A: %08x %d %p\n", handle
, count
, buffer
);
336 if (!count
) return 0;
338 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
339 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
341 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
347 return BITMAP_GetObject32( (BITMAPOBJ
*)ptr
, count
, buffer
);
352 fprintf( stderr
, "GetObject32: magic %04x not implemented\n",
360 /***********************************************************************
361 * GetObject32W (GDI32.206)
363 INT32
GetObject32W( HANDLE32 handle
, INT32 count
, LPVOID buffer
)
365 return GetObject32A( handle
, count
, buffer
);
369 /***********************************************************************
370 * SelectObject (GDI.45)
372 HANDLE
SelectObject( HDC hdc
, HANDLE handle
)
374 GDIOBJHDR
* ptr
= NULL
;
377 dprintf_gdi(stddeb
, "SelectObject: hdc=%04x %04x\n", hdc
, handle
);
378 if ((handle
>= FIRST_STOCK_HANDLE
) && (handle
<= LAST_STOCK_HANDLE
))
379 ptr
= StockObjects
[handle
- FIRST_STOCK_HANDLE
];
381 ptr
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
384 dc
= (DC
*) GDI_GetObjPtr( hdc
, DC_MAGIC
);
387 dc
= (DC
*)GDI_GetObjPtr(hdc
, METAFILE_DC_MAGIC
);
394 return PEN_SelectObject( dc
, handle
, (PENOBJ
*)ptr
);
396 return BRUSH_SelectObject( dc
, handle
, (BRUSHOBJ
*)ptr
);
398 return BITMAP_SelectObject( dc
, handle
, (BITMAPOBJ
*)ptr
);
400 return FONT_SelectObject( dc
, handle
, (FONTOBJ
*)ptr
);
402 return (HANDLE
)SelectClipRgn( hdc
, handle
);
408 /***********************************************************************
409 * UnrealizeObject (GDI.150)
411 BOOL
UnrealizeObject( HANDLE obj
)
413 /* Check if object is valid */
415 GDIOBJHDR
* header
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( obj
);
416 if (!header
) return FALSE
;
418 dprintf_gdi( stddeb
, "UnrealizeObject: %04x\n", obj
);
420 /* Unrealize object */
422 switch(header
->wMagic
)
425 return PALETTE_UnrealizeObject( obj
, (PALETTEOBJ
*)header
);
428 /* Windows resets the brush origin. We don't need to. */
435 /***********************************************************************
436 * EnumObjects (GDI.71)
438 INT
EnumObjects( HDC hdc
, INT nObjType
, GOBJENUMPROC lpEnumFunc
, LPARAM lParam
)
440 /* Solid colors to enumerate */
441 static const COLORREF solid_colors
[] =
442 { RGB(0x00,0x00,0x00), RGB(0xff,0xff,0xff),
443 RGB(0xff,0x00,0x00), RGB(0x00,0xff,0x00),
444 RGB(0x00,0x00,0xff), RGB(0xff,0xff,0x00),
445 RGB(0xff,0x00,0xff), RGB(0x00,0xff,0xff),
446 RGB(0x80,0x00,0x00), RGB(0x00,0x80,0x00),
447 RGB(0x80,0x80,0x00), RGB(0x00,0x00,0x80),
448 RGB(0x80,0x00,0x80), RGB(0x00,0x80,0x80),
449 RGB(0x80,0x80,0x80), RGB(0xc0,0xc0,0xc0)
454 LOGBRUSH16
*brush
= NULL
;
456 dprintf_gdi( stddeb
, "EnumObjects: %04x %d %08lx %08lx\n",
457 hdc
, nObjType
, (DWORD
)lpEnumFunc
, lParam
);
461 /* Enumerate solid pens */
462 if (!(pen
= SEGPTR_NEW(LOGPEN16
))) break;
463 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
465 pen
->lopnStyle
= PS_SOLID
;
466 pen
->lopnWidth
.x
= 1;
467 pen
->lopnWidth
.y
= 0;
468 pen
->lopnColor
= solid_colors
[i
];
469 retval
= CallEnumObjectsProc( (FARPROC16
)lpEnumFunc
,
470 SEGPTR_GET(pen
), lParam
);
471 dprintf_gdi( stddeb
, "EnumObject: solid pen %08lx, ret=%d\n",
472 solid_colors
[i
], retval
);
479 /* Enumerate solid brushes */
480 if (!(brush
= SEGPTR_NEW(LOGBRUSH16
))) break;
481 for (i
= 0; i
< sizeof(solid_colors
)/sizeof(solid_colors
[0]); i
++)
483 brush
->lbStyle
= BS_SOLID
;
484 brush
->lbColor
= solid_colors
[i
];
486 retval
= CallEnumObjectsProc( (FARPROC16
)lpEnumFunc
,
487 SEGPTR_GET(brush
), lParam
);
488 dprintf_gdi( stddeb
, "EnumObject: solid brush %08lx, ret=%d\n",
489 solid_colors
[i
], retval
);
493 /* Now enumerate hatched brushes */
494 if (retval
) for (i
= HS_HORIZONTAL
; i
<= HS_DIAGCROSS
; i
++)
496 brush
->lbStyle
= BS_HATCHED
;
497 brush
->lbColor
= RGB(0,0,0);
499 retval
= CallEnumObjectsProc( (FARPROC16
)lpEnumFunc
,
500 SEGPTR_GET(brush
), lParam
);
501 dprintf_gdi( stddeb
, "EnumObject: hatched brush %d, ret=%d\n",
509 fprintf( stderr
, "EnumObjects: invalid type %d\n", nObjType
);
516 /***********************************************************************
517 * IsGDIObject (GDI.462)
519 BOOL
IsGDIObject(HANDLE handle
)
523 object
= (GDIOBJHDR
*) GDI_HEAP_LIN_ADDR( handle
);
524 /* FIXME: should check magic here */
525 return (object
!= NULL
);
529 /***********************************************************************
532 INT16
MulDiv16( INT16 foo
, INT16 bar
, INT16 baz
)
535 if (!baz
) return -32768;
536 ret
= (foo
* bar
) / baz
;
537 if ((ret
> 32767) || (ret
< -32767)) return -32768;
542 /***********************************************************************
543 * MulDiv32 (KERNEL32.391)
545 INT32
MulDiv32( INT32 foo
, INT32 bar
, INT32 baz
)
550 ret
= ((long long)foo
* bar
) / baz
;
551 if ((ret
> 2147483647) || (ret
< -2147483647)) return -1;
555 return (foo
* bar
) / baz
;