Sun Feb 18 16:35:54 1996 Alexandre Julliard <julliard@lrc.epfl.ch> * [controls/desktop.c] Look for the wallpaper file in the Windows directory. * [controls/menu.c] Fixed swapped parameters in SetMenuItemBitmaps(). Create a separator in MENU_SetItemData() when the string is NULL. * [file/dosfs.c] DOSFS_FindNext: don't return '.' and '..' in a drive root dir. * [files/file.c] Added a DOS_FILE structure to store per-file information (not really used yet). Fixed _lread and _hread to check the size of the buffer before calling Unix read() to avoid EFAULT error. * [misc/exec.c] Return TRUE in WinHelp() for HELP_QUIT to quiet Notepad on exit. * [miscemu/instr.c] Call DOSMEM_Alarm() in INSTR_ReplaceSelector(). This should fix programs that poll the BIOS counter, provided they reload the selector on every read. * [miscemu/int21.c] Re-implemented FindFirst/FindNext for FCB calls. * [windows/message.c] [windows/winpos.c] Merged MSG_GetWindowForEvent() and WINPOS_WindowFromPoint(). * [windows/nonclient.c] [windows/win.c] [include/windows.h] Added a per-window WIN_MANAGED flag; only windows that have a dialog frame or a sizing border are managed. Sat Feb 17 18:25:00 1996 Thomas Sandford <tdgsandf@prds-grn.demon.co.uk> * [if1632/Makefile.in] Added -g flag to compilation of .c files generated from *32.spec. * [if1632/gdi32.spec] Numerous additional functions implemented. * if1632/user32.spec] wsprintfA maps to vsprintf not wsprintf Numerous additional functions implemented. * [include/gdi.h] [objects/gdiobj.c] New #define MAGIC_DONTCARE added. This is used in GDI_GetObjPtr to enable getting a pointer to a GDI object of unknow type. * [win32/gdi32.c] New file. * [win32/param32.c] WIN32_MoveToEx() - handle NULL pointer argument. * [win32/user32.c] USER32_InvalidateRect - handle passing of a NULL pointer. USER32_SetTimer - New function. * [files/directory.c] Fixed DIR_Init() (off by one in allocation of space for environment variables). * [files/drive.c] Added <sys/types.h> to #includes (prerequisite for <sys/stat.h> on FreeBSD). Fri Feb 16 10:26:56 1996 Andreas Kirschbaum <ank@rbg.informatik.th-darmstadt.de> * [controls/menu.c] Memory leak plugged. * [controls/edit.c] Erase space with function ExtTextOut(). This eliminates the use of xmalloc(). Memory leak in EDIT_WriteText plugged. * [debugger/db_disasm.c] Operand for scas now is di. * [files/profile.c] PROFILE_GetSection was copying too much data. PROFILE_GetSection now returns the correct value. It was returning the number of unused instead of used bytes. * [objects/dc.c] Corrected two typos in comments. * [objects/font.c] FONT_MatchFont didn't return if it couldn't find any font. * [objects/oembitmap.c] Free object only if it has been allocated. * [windows/scroll.c] Memory leak in ScrollDC plugged. Tue Feb 13 11:17:00 1996 William Magro <wmagro@tc.cornell.edu> * [controls/edit.c] Implemented ES_NOHIDESEL style, shift+click selection, shift+{arrow,home,end,pgup,pgdn} selection. Optimized (de)selection drawing. Changed selection drawing to use correct system colors instead of inverting. Fixed deleting or backspacing across a '\r\n' end of line pair. Selection now anchors correctly. Fixed text leaking and extra garbage problem bug uncovered by change in class style in wine960131. * [controls/widgets.c] Class flags now match those of Windows. Mon Feb 12 21:28:19 1996 Martin von Loewis <loewis@informatik.hu-berlin.de> * [controls/widgets.c] WIDGETS_Init: RELAY32_GetEntryPoint does not take a string anymore. * [if1632/Makefile.in][if1632/relay32.c][include/relay32.h] comctl32.spec ole32.spec winspool.spec: new files. RELAY32_Init: call initialization of new DLLs. RELAY32_GetEntryPoint: expects WIN32_builtin* now. RELAY32_MakeFakeModule: new function. * [if1632/gdi32.spec][if1632/kernel32.spec][if1632/user32.spec] Added Win95 functions. Ordinals now differ from both NT and Win95 HeapCreate, CreateDialogIndirectParamA, CreateDialogIndirectParamW, CreateDialogParamA, CreateDialogParamW, DialogBoxIndirectParamA DialogBoxIndirectParamW, DialogBoxParamA, DialogBoxParamW: new relays. * [if1632/shell32.spec] shell32.spec: renumbered all functions to take into account ordinals. These seem to be identical between NT and Win95. * [include/dialog.h][windows/dialog.c] xBaseUnit,yBaseUnit,DIALOG_DoDialogBox: made non-static. * [include/handle32.h] New handle types VRANGE, HEAP, HEAPITEM. * [include/pe_image.h][loader/pe_image.c] struct w_files: new field builtin. PE_FindExportedFunction: support ordinals. PE_GetProcAddress: call RELAY32_GetEntryPoint for builtins. fixup_imports: support ordinals. PE_LoadImage: prefer directories over segments. * [include/resource.h][win32/resource.c] FindResource32: changed parameter from LPCTSTR to LPCWSTR check LANG_NEUTRAL if LANG_ENGLISH fails. LoadAcceleratorsW,SizeofResource32,AccessResource32: disabled because it's broken. Casted to and from LPWSTR at various places. * [include/string32.h][win32/string32.c] Changed prototypes to take const arguments where appropriate. * [include/struct32.h] New structures DLGTEMPLATE32, DLGITEMTEMPLATE32. * [tools/build.c] BuildSpec32Files: generate Base value into code, generate call to RELAY32_MakeFakeModule. * [win32/heap.c] This is still not finished and needs rework. HeapAlloc: renamed to SIMPLE_HeapAlloc, implemented HeapAlloc. HeapCreate: implemented on top of VirtualAlloc, which does not work yet HeapDestroy, HEAP_GrowHeap, HeapFree: new functions. * [win32/memory.c] Support for VRANGE_OBJECT. This is not yet called from any place, and needs more platform specific support MEMORY_FindVrange, MEMORY_IsVrangeFree, MEMORY_InsertVrange, MEMORY_AllocVrange, MEMORY_ReleaseVrange: new functions. * [win32/user32.c] WIN32_CreateWindowExA: don't GlobalAlloc for integer class and window names, as in dialogs. Implemented dialog functions (see user32.spec). * [windows/caret.c] CARET_Initialize: call RELAY32_GetBuiltinDLL. Mon Feb 12 18:52:40 1996 Jim Peterson <jspeter@birch.ee.vt.edu> * [controls/edit.c] Removed commented out #ifdefs for WINELIB. * [tools/makehtml.pl] Put in error checking when trying to open a file. * [libtest/Makefile.in] [libtest/new.c] [libtest/hello4.c] Added two new targets: hello4 and new. * [include/windows.h] Added definition of DEVMODE structure, although it's not yet used. Modified various API functions from CreateDC() to Escape(), in order to make them more compliant with the strict API definitions. * [include/wintypes.h] Added 'typedef char TCHAR'. It probably should be defined as 'short', but then we would have to support such characters. Also did 'typedef const TCHAR* LPCTSTR' and 'typedef TCHAR* LPTSTR'. Also defined WNDENUMPROC, FONTENUMPROC, GOBJENUMPROC, PROPENUMPROC MFENUMPROC, and HGDIOBJ. Mon Feb 5 16:42:07 1996 Frans van Dorsselaer <dorssel@rulhm1.leidenuniv.nl> * [misc/commdlg.c] Patched a bug that occurred in the internal COMMDLG module for the FileOpen(), FileSave() and FileSaveAs() functions. The file-type combobox is now handled correctly. Fri Feb 2 22:52:58 1996 Roman Dolejsi <roman@sorry.vse.cz> * [resources/sysres_Cz.rc] Added support for Czech [Cz] language. Thu Feb 1 00:35:04 1996 Philippe De Muyter <phdm@info.ucl.ac.be> * [objects/font.c] FONT_matchfont : for fixed-spacing fonts, allow 'c' if 'm' fails; for variable-spacing fonts : allow '*' if 'p' fails; if asked lfHeight is -1, assume 0. CreateFontIndirect : if font parameter is NULL, issue an error message. CreateFont : null-terminate lfFaceName. ParseFontParms : debug code turned off : too verbose. InitFontsList : recognize *-c-* fonts as fixed-spacing fonts. * [objects/color.c] ColorToPhysical : admit 0xff...... COLORREF's as 0x00...... ones.
561 lines
16 KiB
C
561 lines
16 KiB
C
/*
|
|
* GDI functions
|
|
*
|
|
* Copyright 1993 Alexandre Julliard
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include "gdi.h"
|
|
#include "color.h"
|
|
#include "bitmap.h"
|
|
#include "brush.h"
|
|
#include "font.h"
|
|
#include "palette.h"
|
|
#include "pen.h"
|
|
#include "region.h"
|
|
#include "callback.h"
|
|
#include "stddebug.h"
|
|
/* #define DEBUG_GDI */
|
|
#include "debug.h"
|
|
#include "xmalloc.h"
|
|
|
|
LPSTR GDI_Heap = NULL;
|
|
WORD GDI_HeapSel = 0;
|
|
|
|
/* Object types for EnumObjects() */
|
|
#define OBJ_PEN 1
|
|
#define OBJ_BRUSH 2
|
|
|
|
#define MAX_OBJ 1024
|
|
HANDLE *lpPenBrushList = NULL;
|
|
|
|
|
|
/***********************************************************************
|
|
* GDI stock objects
|
|
*/
|
|
|
|
static BRUSHOBJ WhiteBrush =
|
|
{
|
|
{ 0, BRUSH_MAGIC, 1, 0 }, /* header */
|
|
{ BS_SOLID, RGB(255,255,255), 0 } /* logbrush */
|
|
};
|
|
|
|
static BRUSHOBJ LtGrayBrush =
|
|
{
|
|
{ 0, BRUSH_MAGIC, 1, 0 }, /* header */
|
|
{ BS_SOLID, RGB(192,192,192), 0 } /* logbrush */
|
|
};
|
|
|
|
static BRUSHOBJ GrayBrush =
|
|
{
|
|
{ 0, BRUSH_MAGIC, 1, 0 }, /* header */
|
|
{ BS_SOLID, RGB(128,128,128), 0 } /* logbrush */
|
|
};
|
|
|
|
static BRUSHOBJ DkGrayBrush =
|
|
{
|
|
{ 0, BRUSH_MAGIC, 1, 0 }, /* header */
|
|
{ BS_SOLID, RGB(64,64,64), 0 } /* logbrush */
|
|
};
|
|
|
|
static BRUSHOBJ BlackBrush =
|
|
{
|
|
{ 0, BRUSH_MAGIC, 1, 0 }, /* header */
|
|
{ BS_SOLID, RGB(0,0,0), 0 } /* logbrush */
|
|
};
|
|
|
|
static BRUSHOBJ NullBrush =
|
|
{
|
|
{ 0, BRUSH_MAGIC, 1, 0 }, /* header */
|
|
{ BS_NULL, 0, 0 } /* logbrush */
|
|
};
|
|
|
|
static PENOBJ WhitePen =
|
|
{
|
|
{ 0, PEN_MAGIC, 1, 0 }, /* header */
|
|
{ PS_SOLID, { 1, 0 }, RGB(255,255,255) } /* logpen */
|
|
};
|
|
|
|
static PENOBJ BlackPen =
|
|
{
|
|
{ 0, PEN_MAGIC, 1, 0 }, /* header */
|
|
{ PS_SOLID, { 1, 0 }, RGB(0,0,0) } /* logpen */
|
|
};
|
|
|
|
static PENOBJ NullPen =
|
|
{
|
|
{ 0, PEN_MAGIC, 1, 0 }, /* header */
|
|
{ PS_NULL, { 1, 0 }, 0 } /* logpen */
|
|
};
|
|
|
|
static FONTOBJ OEMFixedFont =
|
|
{
|
|
{ 0, FONT_MAGIC, 1, 0 }, /* header */
|
|
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, OEM_CHARSET,
|
|
0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN, "" }
|
|
};
|
|
|
|
static FONTOBJ AnsiFixedFont =
|
|
{
|
|
{ 0, FONT_MAGIC, 1, 0 }, /* header */
|
|
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
|
|
0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN, "" }
|
|
};
|
|
|
|
static FONTOBJ AnsiVarFont =
|
|
{
|
|
{ 0, FONT_MAGIC, 1, 0 }, /* header */
|
|
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
|
|
0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS, "" }
|
|
};
|
|
|
|
static FONTOBJ SystemFont =
|
|
{
|
|
{ 0, FONT_MAGIC, 1, 0 }, /* header */
|
|
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
|
|
0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS, "" }
|
|
};
|
|
|
|
static FONTOBJ DeviceDefaultFont =
|
|
{
|
|
{ 0, FONT_MAGIC, 1, 0 }, /* header */
|
|
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
|
|
0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS, "" }
|
|
};
|
|
|
|
static FONTOBJ SystemFixedFont =
|
|
{
|
|
{ 0, FONT_MAGIC, 1, 0 }, /* header */
|
|
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
|
|
0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN, "" }
|
|
};
|
|
|
|
|
|
static GDIOBJHDR * StockObjects[NB_STOCK_OBJECTS] =
|
|
{
|
|
(GDIOBJHDR *) &WhiteBrush,
|
|
(GDIOBJHDR *) &LtGrayBrush,
|
|
(GDIOBJHDR *) &GrayBrush,
|
|
(GDIOBJHDR *) &DkGrayBrush,
|
|
(GDIOBJHDR *) &BlackBrush,
|
|
(GDIOBJHDR *) &NullBrush,
|
|
(GDIOBJHDR *) &WhitePen,
|
|
(GDIOBJHDR *) &BlackPen,
|
|
(GDIOBJHDR *) &NullPen,
|
|
NULL,
|
|
(GDIOBJHDR *) &OEMFixedFont,
|
|
(GDIOBJHDR *) &AnsiFixedFont,
|
|
(GDIOBJHDR *) &AnsiVarFont,
|
|
(GDIOBJHDR *) &SystemFont,
|
|
(GDIOBJHDR *) &DeviceDefaultFont,
|
|
NULL, /* DEFAULT_PALETTE created by COLOR_Init */
|
|
(GDIOBJHDR *) &SystemFixedFont
|
|
};
|
|
|
|
|
|
/***********************************************************************
|
|
* GDI_Init
|
|
*
|
|
* GDI initialisation.
|
|
*/
|
|
BOOL GDI_Init(void)
|
|
{
|
|
HPALETTE hpalette;
|
|
|
|
#ifndef WINELIB
|
|
/* Create GDI heap */
|
|
|
|
if (!(GDI_HeapSel = GlobalAlloc(GMEM_FIXED, GDI_HEAP_SIZE))) return FALSE;
|
|
GDI_Heap = GlobalLock( GDI_HeapSel );
|
|
LocalInit( GDI_HeapSel, 0, GDI_HEAP_SIZE-1 );
|
|
#endif
|
|
|
|
/* Create default palette */
|
|
|
|
if (!(hpalette = COLOR_Init())) return FALSE;
|
|
StockObjects[DEFAULT_PALETTE] = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( hpalette );
|
|
|
|
/* Create default bitmap */
|
|
|
|
if (!BITMAP_Init()) return FALSE;
|
|
|
|
/* Initialise brush dithering */
|
|
|
|
if (!BRUSH_Init()) return FALSE;
|
|
|
|
/* Initialise fonts */
|
|
|
|
if (!FONT_Init()) return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GDI_AllocObject
|
|
*/
|
|
HANDLE GDI_AllocObject( WORD size, WORD magic )
|
|
{
|
|
static DWORD count = 0;
|
|
GDIOBJHDR * obj;
|
|
HANDLE handle = GDI_HEAP_ALLOC( size );
|
|
if (!handle) return 0;
|
|
obj = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( handle );
|
|
obj->hNext = 0;
|
|
obj->wMagic = magic;
|
|
obj->dwCount = ++count;
|
|
return handle;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GDI_FreeObject
|
|
*/
|
|
BOOL GDI_FreeObject( HANDLE handle )
|
|
{
|
|
GDIOBJHDR * object;
|
|
|
|
/* Can't free stock objects */
|
|
if ((DWORD)handle >= FIRST_STOCK_HANDLE &&
|
|
(DWORD)handle <= LAST_STOCK_HANDLE ) return TRUE;
|
|
|
|
object = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( handle );
|
|
if (!object) return FALSE;
|
|
object->wMagic = 0; /* Mark it as invalid */
|
|
|
|
/* Free object */
|
|
|
|
GDI_HEAP_FREE( handle );
|
|
return TRUE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* GDI_GetObjPtr
|
|
*
|
|
* Return a pointer to the GDI object associated to the handle.
|
|
* Return NULL if the object has the wrong magic number.
|
|
*/
|
|
GDIOBJHDR * GDI_GetObjPtr( HANDLE handle, WORD magic )
|
|
{
|
|
GDIOBJHDR * ptr = NULL;
|
|
|
|
if ((DWORD)handle >= FIRST_STOCK_HANDLE &&
|
|
(DWORD)handle <= LAST_STOCK_HANDLE )
|
|
ptr = StockObjects[(DWORD)handle - FIRST_STOCK_HANDLE];
|
|
else
|
|
ptr = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( handle );
|
|
if (!ptr) return NULL;
|
|
if ((magic != MAGIC_DONTCARE) && (ptr->wMagic != magic)) return NULL;
|
|
return ptr;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* DeleteObject (GDI.69)
|
|
*/
|
|
BOOL DeleteObject( HGDIOBJ obj )
|
|
{
|
|
/* Check if object is valid */
|
|
|
|
GDIOBJHDR * header = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( obj );
|
|
if (!header) return FALSE;
|
|
|
|
dprintf_gdi(stddeb, "DeleteObject: "NPFMT"\n", obj );
|
|
|
|
/* Delete object */
|
|
|
|
switch(header->wMagic)
|
|
{
|
|
case PEN_MAGIC: return GDI_FreeObject( obj );
|
|
case BRUSH_MAGIC: return BRUSH_DeleteObject( obj, (BRUSHOBJ*)header );
|
|
case FONT_MAGIC: return GDI_FreeObject( obj );
|
|
case PALETTE_MAGIC: return GDI_FreeObject( obj );
|
|
case BITMAP_MAGIC: return BITMAP_DeleteObject( obj, (BITMAPOBJ*)header);
|
|
case REGION_MAGIC: return REGION_DeleteObject( obj, (RGNOBJ*)header );
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetStockObject (GDI.87)
|
|
*/
|
|
HANDLE GetStockObject( int obj )
|
|
{
|
|
if ((obj < 0) || (obj >= NB_STOCK_OBJECTS)) return 0;
|
|
if (!StockObjects[obj]) return 0;
|
|
dprintf_gdi(stddeb, "GetStockObject: returning %ld\n",
|
|
(DWORD)FIRST_STOCK_HANDLE + obj );
|
|
return (HANDLE)((DWORD)FIRST_STOCK_HANDLE + obj);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* GetObject (GDI.82)
|
|
*/
|
|
int GetObject( HANDLE handle, int count, LPSTR buffer )
|
|
{
|
|
GDIOBJHDR * ptr = NULL;
|
|
dprintf_gdi(stddeb, "GetObject: "NPFMT" %d %p\n", handle, count, buffer );
|
|
if (!count) return 0;
|
|
|
|
if ((DWORD)handle >= FIRST_STOCK_HANDLE &&
|
|
(DWORD)handle <= LAST_STOCK_HANDLE )
|
|
ptr = StockObjects[(DWORD)handle - FIRST_STOCK_HANDLE];
|
|
else
|
|
ptr = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( handle );
|
|
if (!ptr) return 0;
|
|
|
|
switch(ptr->wMagic)
|
|
{
|
|
case PEN_MAGIC:
|
|
return PEN_GetObject( (PENOBJ *)ptr, count, buffer );
|
|
case BRUSH_MAGIC:
|
|
return BRUSH_GetObject( (BRUSHOBJ *)ptr, count, buffer );
|
|
case BITMAP_MAGIC:
|
|
return BITMAP_GetObject( (BITMAPOBJ *)ptr, count, buffer );
|
|
case FONT_MAGIC:
|
|
return FONT_GetObject( (FONTOBJ *)ptr, count, buffer );
|
|
case PALETTE_MAGIC:
|
|
return PALETTE_GetObject( (PALETTEOBJ *)ptr, count, buffer );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* SelectObject (GDI.45)
|
|
*/
|
|
HANDLE SelectObject( HDC hdc, HANDLE handle )
|
|
{
|
|
GDIOBJHDR * ptr = NULL;
|
|
DC * dc;
|
|
|
|
dprintf_gdi(stddeb, "SelectObject: "NPFMT" "NPFMT"\n", hdc, handle );
|
|
if ((DWORD)handle >= FIRST_STOCK_HANDLE &&
|
|
(DWORD)handle <= LAST_STOCK_HANDLE )
|
|
ptr = StockObjects[(DWORD)handle - FIRST_STOCK_HANDLE];
|
|
else
|
|
ptr = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( handle );
|
|
if (!ptr) return 0;
|
|
|
|
dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC );
|
|
if (!dc)
|
|
{
|
|
dc = (DC *)GDI_GetObjPtr(hdc, METAFILE_DC_MAGIC);
|
|
if (!dc) return 0;
|
|
}
|
|
|
|
switch(ptr->wMagic)
|
|
{
|
|
case PEN_MAGIC:
|
|
return PEN_SelectObject( dc, handle, (PENOBJ *)ptr );
|
|
case BRUSH_MAGIC:
|
|
return BRUSH_SelectObject( hdc, dc, handle, (BRUSHOBJ *)ptr );
|
|
case BITMAP_MAGIC:
|
|
return BITMAP_SelectObject( hdc, dc, handle, (BITMAPOBJ *)ptr );
|
|
case FONT_MAGIC:
|
|
return FONT_SelectObject( dc, handle, (FONTOBJ *)ptr );
|
|
case REGION_MAGIC:
|
|
return (HANDLE)SelectClipRgn( hdc, handle );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* UnrealizeObject (GDI.150)
|
|
*/
|
|
BOOL UnrealizeObject( HANDLE handle )
|
|
{
|
|
dprintf_gdi(stdnimp, "UnrealizeObject: "NPFMT"\n", handle );
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* EnumObjects (GDI.71)
|
|
*/
|
|
INT EnumObjects( HDC hdc, INT nObjType, GOBJENUMPROC lpEnumFunc, LPARAM lParam )
|
|
{
|
|
/* Solid colors to enumerate */
|
|
static const COLORREF solid_colors[] =
|
|
{ RGB(0x00,0x00,0x00), RGB(0xff,0xff,0xff),
|
|
RGB(0xff,0x00,0x00), RGB(0x00,0xff,0x00),
|
|
RGB(0x00,0x00,0xff), RGB(0xff,0xff,0x00),
|
|
RGB(0xff,0x00,0xff), RGB(0x00,0xff,0xff),
|
|
RGB(0x80,0x00,0x00), RGB(0x00,0x80,0x00),
|
|
RGB(0x80,0x80,0x00), RGB(0x00,0x00,0x80),
|
|
RGB(0x80,0x00,0x80), RGB(0x00,0x80,0x80),
|
|
RGB(0x80,0x80,0x80), RGB(0xc0,0xc0,0xc0)
|
|
};
|
|
|
|
int i, retval = 0;
|
|
|
|
dprintf_gdi( stddeb, "EnumObjects: "NPFMT" %d %08lx %08lx\n",
|
|
hdc, nObjType, (DWORD)lpEnumFunc, lParam );
|
|
switch(nObjType)
|
|
{
|
|
case OBJ_PEN:
|
|
/* Enumerate solid pens */
|
|
for (i = 0; i < sizeof(solid_colors)/sizeof(solid_colors[0]); i++)
|
|
{
|
|
LOGPEN pen = { PS_SOLID, { 1, 0 }, solid_colors[i] };
|
|
retval = CallEnumObjectsProc( lpEnumFunc, MAKE_SEGPTR(&pen),
|
|
lParam );
|
|
dprintf_gdi( stddeb, "EnumObject: solid pen %08lx, ret=%d\n",
|
|
solid_colors[i], retval);
|
|
if (!retval) break;
|
|
}
|
|
break;
|
|
|
|
case OBJ_BRUSH:
|
|
/* Enumerate solid brushes */
|
|
for (i = 0; i < sizeof(solid_colors)/sizeof(solid_colors[0]); i++)
|
|
{
|
|
LOGBRUSH brush = { BS_SOLID, solid_colors[i], 0 };
|
|
retval = CallEnumObjectsProc( lpEnumFunc, MAKE_SEGPTR(&brush),
|
|
lParam );
|
|
dprintf_gdi( stddeb, "EnumObject: solid brush %08lx, ret=%d\n",
|
|
solid_colors[i], retval);
|
|
if (!retval) break;
|
|
}
|
|
if (!retval) break;
|
|
|
|
/* Now enumerate hatched brushes */
|
|
for (i = HS_HORIZONTAL; i <= HS_DIAGCROSS; i++)
|
|
{
|
|
LOGBRUSH brush = { BS_HATCHED, RGB(0,0,0), i };
|
|
retval = CallEnumObjectsProc( lpEnumFunc, MAKE_SEGPTR(&brush),
|
|
lParam );
|
|
dprintf_gdi( stddeb, "EnumObject: hatched brush %d, ret=%d\n",
|
|
i, retval);
|
|
if (!retval) break;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
fprintf( stderr, "EnumObjects: invalid type %d\n", nObjType );
|
|
break;
|
|
}
|
|
return retval;
|
|
#if 0
|
|
|
|
/* HANDLE handle;
|
|
DC *dc;*/
|
|
HANDLE *lphObj;
|
|
GDIOBJHDR *header;
|
|
WORD wMagic;
|
|
LPSTR lpLog; /* Point to a LOGBRUSH or LOGPEN struct */
|
|
HANDLE hLog;
|
|
int nRet = 0;
|
|
|
|
if (lpEnumFunc == 0) {
|
|
fprintf(stderr,"EnumObjects // Bad EnumProc callback address !\n");
|
|
return 0;
|
|
}
|
|
switch (nObjType) {
|
|
case OBJ_PEN:
|
|
wMagic = PEN_MAGIC;
|
|
dprintf_gdi(stddeb,"EnumObjects("NPFMT", OBJ_PEN, %08lx, %p);\n",
|
|
hDC, (LONG)lpEnumFunc, lpData);
|
|
hLog = GDI_HEAP_ALLOC( sizeof(LOGPEN) );
|
|
lpLog = (LPSTR) GDI_HEAP_LIN_ADDR(hLog);
|
|
if (lpLog == NULL) {
|
|
fprintf(stderr,"EnumObjects // Unable to alloc LOGPEN struct !\n");
|
|
return 0;
|
|
}
|
|
break;
|
|
case OBJ_BRUSH:
|
|
wMagic = BRUSH_MAGIC;
|
|
dprintf_gdi(stddeb,"EnumObjects("NPFMT", OBJ_BRUSH, %08lx, %p);\n",
|
|
hDC, (LONG)lpEnumFunc, lpData);
|
|
hLog = GDI_HEAP_ALLOC( sizeof(LOGBRUSH) );
|
|
lpLog = (LPSTR) GDI_HEAP_LIN_ADDR(hLog);
|
|
if (lpLog == NULL) {
|
|
fprintf(stderr,"EnumObjects // Unable to alloc LOGBRUSH struct !\n");
|
|
return 0;
|
|
}
|
|
break;
|
|
default:
|
|
fprintf(stderr,"EnumObjects("NPFMT", %04X, %08lx, %p); // Unknown OBJ type !\n",
|
|
hDC, nObjType, (LONG)lpEnumFunc, lpData);
|
|
return 0;
|
|
}
|
|
#ifdef notdef /* FIXME: stock object ptr won't work in callback */
|
|
dprintf_gdi(stddeb,"EnumObjects // Stock Objects first !\n");
|
|
for (i = 0; i < NB_STOCK_OBJECTS; i++) {
|
|
header = StockObjects[i];
|
|
if (header->wMagic == wMagic) {
|
|
PEN_GetObject( (PENOBJ *)header, sizeof(LOGPEN), lpLog);
|
|
BRUSH_GetObject( (BRUSHOBJ *)header, sizeof(LOGBRUSH),lpLog);
|
|
dprintf_gdi(stddeb,"EnumObjects // StockObj lpLog=%p lpData=%p\n", lpLog, lpData);
|
|
if (header->wMagic == BRUSH_MAGIC) {
|
|
dprintf_gdi(stddeb,"EnumObjects // StockBrush lbStyle=%04X\n", ((LPLOGBRUSH)lpLog)->lbStyle);
|
|
dprintf_gdi(stddeb,"EnumObjects // StockBrush lbColor=%08lX\n", ((LPLOGBRUSH)lpLog)->lbColor);
|
|
dprintf_gdi(stddeb,"EnumObjects // StockBrush lbHatch=%04X\n", ((LPLOGBRUSH)lpLog)->lbHatch);
|
|
}
|
|
if (header->wMagic == PEN_MAGIC) {
|
|
dprintf_gdi(stddeb,"EnumObjects // StockPen lopnStyle=%04X\n", ((LPLOGPEN)lpLog)->lopnStyle);
|
|
dprintf_gdi(stddeb,"EnumObjects // StockPen lopnWidth=%d\n", ((LPLOGPEN)lpLog)->lopnWidth.x);
|
|
dprintf_gdi(stddeb,"EnumObjects // StockPen lopnColor=%08lX\n", ((LPLOGPEN)lpLog)->lopnColor);
|
|
}
|
|
nRet = CallEnumObjectsProc( lpEnumFunc,
|
|
GDI_HEAP_SEG_ADDR(hLog),
|
|
(int)lpData );
|
|
dprintf_gdi(stddeb,"EnumObjects // after Callback!\n");
|
|
if (nRet == 0) {
|
|
GDI_HEAP_FREE(hLog);
|
|
dprintf_gdi(stddeb,"EnumObjects // EnumEnd requested by application !\n");
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
dprintf_gdi(stddeb,"EnumObjects // Now DC owned objects %p !\n", header);
|
|
#endif /* notdef */
|
|
|
|
if (lpPenBrushList == NULL) return 0;
|
|
for (lphObj = lpPenBrushList; *lphObj != 0; ) {
|
|
dprintf_gdi(stddeb,"EnumObjects // *lphObj="NPFMT"\n", *lphObj);
|
|
header = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR(*lphObj++);
|
|
if (header->wMagic == wMagic) {
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Obj lpLog=%p lpData=%p\n", lpLog, lpData);
|
|
if (header->wMagic == BRUSH_MAGIC) {
|
|
BRUSH_GetObject( (BRUSHOBJ *)header, sizeof(LOGBRUSH), lpLog);
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Brush lbStyle=%04X\n", ((LPLOGBRUSH)lpLog)->lbStyle);
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Brush lbColor=%08lX\n", ((LPLOGBRUSH)lpLog)->lbColor);
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Brush lbHatch=%04lX\n", (LONG)((LPLOGBRUSH)lpLog)->lbHatch);
|
|
}
|
|
if (header->wMagic == PEN_MAGIC) {
|
|
PEN_GetObject( (PENOBJ *)header, sizeof(LOGPEN), lpLog);
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Pen lopnStyle=%04X\n", ((LPLOGPEN)lpLog)->lopnStyle);
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Pen lopnWidth=%ld\n", (LONG)((LPLOGPEN)lpLog)->lopnWidth.x);
|
|
dprintf_gdi(stddeb,"EnumObjects // DC_Pen lopnColor=%08lX\n", ((LPLOGPEN)lpLog)->lopnColor);
|
|
}
|
|
nRet = CallEnumObjectsProc(lpEnumFunc, GDI_HEAP_SEG_ADDR(hLog),
|
|
(LONG)lpData);
|
|
if (nRet == 0)
|
|
break;
|
|
}
|
|
}
|
|
GDI_HEAP_FREE(hLog);
|
|
dprintf_gdi(stddeb,"EnumObjects // End of enumeration !\n");
|
|
return nRet;
|
|
#endif
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* IsGDIObject(GDI.462)
|
|
*/
|
|
BOOL IsGDIObject(HANDLE handle)
|
|
{
|
|
GDIOBJHDR *object;
|
|
|
|
object = (GDIOBJHDR *) GDI_HEAP_LIN_ADDR( handle );
|
|
if (object)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|