gdiobj.c 43.3 KB
Newer Older
Alexandre Julliard's avatar
Alexandre Julliard committed
1 2 3 4
/*
 * GDI functions
 *
 * Copyright 1993 Alexandre Julliard
5 6 7 8 9 10 11 12 13 14 15 16 17
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
18
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Alexandre Julliard's avatar
Alexandre Julliard committed
19
 */
Alexandre Julliard's avatar
Alexandre Julliard committed
20

21 22
#include "config.h"

23
#include <assert.h>
Alexandre Julliard's avatar
Alexandre Julliard committed
24
#include <stdlib.h>
25
#include <stdarg.h>
26
#include <stdio.h>
27

28
#include "windef.h"
29
#include "winbase.h"
30
#include "wingdi.h"
31
#include "winreg.h"
32
#include "winerror.h"
33
#include "winternl.h"
34

35
#include "gdi_private.h"
36
#include "wine/debug.h"
Alexandre Julliard's avatar
Alexandre Julliard committed
37

38
WINE_DEFAULT_DEBUG_CHANNEL(gdi);
39

40
#define HGDIOBJ_32(h16)   ((HGDIOBJ)(ULONG_PTR)(h16))
41

42 43
#define GDI_HEAP_SIZE 0xffe0

Alexandre Julliard's avatar
Alexandre Julliard committed
44
/***********************************************************************
45
 *          GDI stock objects
Alexandre Julliard's avatar
Alexandre Julliard committed
46 47
 */

48 49 50
static const LOGBRUSH WhiteBrush = { BS_SOLID, RGB(255,255,255), 0 };
static const LOGBRUSH BlackBrush = { BS_SOLID, RGB(0,0,0), 0 };
static const LOGBRUSH NullBrush  = { BS_NULL, 0, 0 };
Alexandre Julliard's avatar
Alexandre Julliard committed
51

52 53
static const LOGBRUSH LtGrayBrush = { BS_SOLID, RGB(192,192,192), 0 };
static const LOGBRUSH GrayBrush   = { BS_SOLID, RGB(128,128,128), 0 };
54
static const LOGBRUSH DkGrayBrush = { BS_SOLID, RGB(64,64,64), 0 };
Alexandre Julliard's avatar
Alexandre Julliard committed
55

56 57 58
static const LOGPEN WhitePen = { PS_SOLID, { 0, 0 }, RGB(255,255,255) };
static const LOGPEN BlackPen = { PS_SOLID, { 0, 0 }, RGB(0,0,0) };
static const LOGPEN NullPen  = { PS_NULL,  { 0, 0 }, 0 };
Alexandre Julliard's avatar
Alexandre Julliard committed
59

60 61
static const LOGBRUSH DCBrush = { BS_SOLID, RGB(255,255,255), 0 };
static const LOGPEN DCPen     = { PS_SOLID, { 0, 0 }, RGB(0,0,0) };
62 63 64 65 66 67 68

/* reserve one extra entry for the stock default bitmap */
/* this is what Windows does too */
#define NB_STOCK_OBJECTS (STOCK_LAST+2)

static HGDIOBJ stock_objects[NB_STOCK_OBJECTS];

69 70 71 72 73
static SYSLEVEL GDI_level;
static CRITICAL_SECTION_DEBUG critsect_debug =
{
    0, 0, &GDI_level.crst,
    { &critsect_debug.ProcessLocksList, &critsect_debug.ProcessLocksList },
74
      0, 0, { (DWORD_PTR)(__FILE__ ": GDI_level") }
75 76 77
};
static SYSLEVEL GDI_level = { { &critsect_debug, -1, 0, 0, 0, 0 }, 3 };

78 79 80 81 82 83 84

/****************************************************************************
 *
 *	language-independent stock fonts
 *
 */

85 86 87
static const LOGFONTW OEMFixedFont =
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, OEM_CHARSET,
  0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN, {'\0'} };
Alexandre Julliard's avatar
Alexandre Julliard committed
88

89 90 91
static const LOGFONTW AnsiFixedFont =
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
  0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN, {'\0'} };
Alexandre Julliard's avatar
Alexandre Julliard committed
92

93 94 95 96
static const LOGFONTW AnsiVarFont =
{ 12, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
  0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
  {'M','S',' ','S','a','n','s',' ','S','e','r','i','f','\0'} };
Alexandre Julliard's avatar
Alexandre Julliard committed
97

98 99 100 101 102 103 104 105 106 107 108
/******************************************************************************
 *
 *      language-dependent stock fonts
 *
 *      'ANSI' charset and 'DEFAULT' charset is not same.
 *      The chars in CP_ACP should be drawn with 'DEFAULT' charset.
 *      'ANSI' charset seems to be identical with ISO-8859-1.
 *      'DEFAULT' charset is a language-dependent charset.
 *
 *      'System' font seems to be an alias for language-dependent font.
 */
Alexandre Julliard's avatar
Alexandre Julliard committed
109

110
/*
Vitaly Lipatov's avatar
Vitaly Lipatov committed
111 112 113
 * language-dependent stock fonts for all known charsets
 * please see TranslateCharsetInfo (dlls/gdi/font.c) and
 * CharsetBindingInfo (dlls/x11drv/xfont.c),
114 115 116 117 118 119 120 121
 * and modify entries for your language if needed.
 */
struct DefaultFontInfo
{
        UINT            charset;
        LOGFONTW        SystemFont;
        LOGFONTW        DeviceDefaultFont;
        LOGFONTW        SystemFixedFont;
122
        LOGFONTW        DefaultGuiFont; /* Note for this font the lfHeight member should be the point size */
123 124 125 126 127 128
};

static const struct DefaultFontInfo default_fonts[] =
{
    {   ANSI_CHARSET,
        { /* System */
129
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
130 131 132 133 134 135 136 137 138 139 140 141 142 143
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
144
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ANSI_CHARSET,
145
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
146
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
147 148 149 150
        },
    },
    {   EASTEUROPE_CHARSET,
        { /* System */
151
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, EASTEUROPE_CHARSET,
152 153 154 155 156 157 158 159 160 161 162 163 164 165
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, EASTEUROPE_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, EASTEUROPE_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
166
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, EASTEUROPE_CHARSET,
167
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
168
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
169 170 171 172
        },
    },
    {   RUSSIAN_CHARSET,
        { /* System */
173
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, RUSSIAN_CHARSET,
174 175 176 177 178 179 180 181 182 183 184 185 186 187
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, RUSSIAN_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, RUSSIAN_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
188
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, RUSSIAN_CHARSET,
189
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
190
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
191 192 193 194
        },
    },
    {   GREEK_CHARSET,
        { /* System */
195
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GREEK_CHARSET,
196 197 198 199 200 201 202 203 204 205 206 207 208 209
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GREEK_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GREEK_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
210
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GREEK_CHARSET,
211
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
212
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
213 214 215 216
        },
    },
    {   TURKISH_CHARSET,
        { /* System */
217
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, TURKISH_CHARSET,
218 219 220 221 222 223 224 225 226 227 228 229 230 231
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, TURKISH_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, TURKISH_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
232
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, TURKISH_CHARSET,
233
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
234
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
235 236 237 238
        },
    },
    {   HEBREW_CHARSET,
        { /* System */
239
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HEBREW_CHARSET,
240 241 242 243 244 245 246 247 248 249 250 251 252 253
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HEBREW_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HEBREW_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
254
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HEBREW_CHARSET,
255
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
256
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
257 258 259 260
        },
    },
    {   ARABIC_CHARSET,
        { /* System */
261
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ARABIC_CHARSET,
262 263 264 265 266 267 268 269 270 271 272 273 274 275
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ARABIC_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ARABIC_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
276
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, ARABIC_CHARSET,
277
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
278
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
279 280 281 282
        },
    },
    {   BALTIC_CHARSET,
        { /* System */
283
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, BALTIC_CHARSET,
284 285 286 287 288 289 290 291 292 293 294 295 296 297
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, BALTIC_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, BALTIC_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
298
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, BALTIC_CHARSET,
299
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
300
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
301 302 303 304
        },
    },
    {   THAI_CHARSET,
        { /* System */
305
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, THAI_CHARSET,
306 307 308 309 310 311 312 313 314 315 316 317 318 319
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, THAI_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, THAI_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
320
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, THAI_CHARSET,
321
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
322
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
323 324 325 326
        },
    },
    {   SHIFTJIS_CHARSET,
        { /* System */
327
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, SHIFTJIS_CHARSET,
328 329 330 331 332 333 334 335 336 337 338 339 340 341
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, SHIFTJIS_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, SHIFTJIS_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
342
           9, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, SHIFTJIS_CHARSET,
343
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
344
           {'M','S',' ','U','I',' ','G','o','t','h','i','c','\0'}
345 346 347 348
        },
    },
    {   GB2312_CHARSET,
        { /* System */
349
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GB2312_CHARSET,
350 351 352 353 354 355 356 357 358 359 360 361 362 363
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GB2312_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GB2312_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
364
           9, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, GB2312_CHARSET,
365 366 367 368 369 370
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'M','S',' ','S','o','n','g','\0'}   /* FIXME: Is this correct? */
        },
    },
    {   HANGEUL_CHARSET,
        { /* System */
371
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HANGEUL_CHARSET,
372 373 374 375 376 377 378 379 380 381 382 383 384 385
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HANGEUL_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HANGEUL_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
386
           9, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, HANGEUL_CHARSET,
387
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
388
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
389 390 391 392
        },
    },
    {   CHINESEBIG5_CHARSET,
        { /* System */
393
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, CHINESEBIG5_CHARSET,
394 395 396 397 398 399 400 401 402 403 404 405 406 407
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, CHINESEBIG5_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, CHINESEBIG5_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
408
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, CHINESEBIG5_CHARSET,
409 410 411 412 413 414
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}       /* FIXME - what is the native font??? */
        },
    },
    {   JOHAB_CHARSET,
        { /* System */
415
          16, 7, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, JOHAB_CHARSET,
416 417 418 419 420 421 422 423 424 425 426 427 428 429
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'S','y','s','t','e','m','\0'}
        },
        { /* Device Default */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, JOHAB_CHARSET,
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
           {'\0'}
        },
        { /* System Fixed */
          16, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, JOHAB_CHARSET,
           0, 0, DEFAULT_QUALITY, FIXED_PITCH | FF_MODERN,
           {'\0'}
        },
        { /* DefaultGuiFont */
430
           8, 0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE, JOHAB_CHARSET,
431
           0, 0, DEFAULT_QUALITY, VARIABLE_PITCH | FF_SWISS,
432
           {'M','S',' ','S','h','e','l','l',' ','D','l','g','\0'}
433 434 435
        },
    },
};
Alexandre Julliard's avatar
Alexandre Julliard committed
436

Alexandre Julliard's avatar
Alexandre Julliard committed
437

438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457
/*************************************************************************
 * __wine_make_gdi_object_system    (GDI32.@)
 *
 * USER has to tell GDI that its system brushes and pens are non-deletable.
 * For a description of the GDI object magics and their flags,
 * see "Undocumented Windows" (wrong about the OBJECT_NOSYSTEM flag, though).
 */
void __wine_make_gdi_object_system( HGDIOBJ handle, BOOL set)
{
    GDIOBJHDR *ptr = GDI_GetObjPtr( handle, MAGIC_DONTCARE );

    /* touch the "system" bit of the wMagic field of a GDIOBJHDR */
    if (set)
        ptr->wMagic &= ~OBJECT_NOSYSTEM;
    else
        ptr->wMagic |= OBJECT_NOSYSTEM;

    GDI_ReleaseObj( handle );
}

458 459 460 461 462
/******************************************************************************
 *      get_default_fonts
 */
static const struct DefaultFontInfo* get_default_fonts(UINT charset)
{
463
        unsigned int n;
Alexandre Julliard's avatar
Alexandre Julliard committed
464

465 466 467 468 469
        for(n=0;n<(sizeof(default_fonts)/sizeof(default_fonts[0]));n++)
        {
                if ( default_fonts[n].charset == charset )
                        return &default_fonts[n];
        }
Alexandre Julliard's avatar
Alexandre Julliard committed
470

471 472 473
        FIXME( "unhandled charset 0x%08x - use ANSI_CHARSET for default stock objects\n", charset );
        return &default_fonts[0];
}
474

475

476 477 478 479 480 481
/******************************************************************************
 *      get_default_charset    (internal)
 *
 * get the language-dependent charset that can handle CP_ACP correctly.
 */
static UINT get_default_charset( void )
482
{
483 484 485 486 487
    CHARSETINFO     csi;
    UINT    uACP;

    uACP = GetACP();
    csi.ciCharset = ANSI_CHARSET;
488
    if ( !TranslateCharsetInfo( ULongToPtr(uACP), &csi, TCI_SRCCODEPAGE ) )
489 490 491 492 493 494
    {
        FIXME( "unhandled codepage %u - use ANSI_CHARSET for default stock objects\n", uACP );
        return ANSI_CHARSET;
    }

    return csi.ciCharset;
495
}
496

497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523
static const WCHAR dpi_key_name[] = {'S','o','f','t','w','a','r','e','\\','F','o','n','t','s','\0'};
static const WCHAR dpi_value_name[] = {'L','o','g','P','i','x','e','l','s','\0'};

/******************************************************************************
 *      get_dpi   (internal)
 *
 * get the dpi from the registry
 */
static DWORD get_dpi( void )
{
    DWORD dpi = 96;
    HKEY hkey;

    if (RegOpenKeyW(HKEY_CURRENT_CONFIG, dpi_key_name, &hkey) == ERROR_SUCCESS)
    {
        DWORD type, size, new_dpi;

        size = sizeof(new_dpi);
        if(RegQueryValueExW(hkey, dpi_value_name, NULL, &type, (void *)&new_dpi, &size) == ERROR_SUCCESS)
        {
            if(type == REG_DWORD && new_dpi != 0)
                dpi = new_dpi;
        }
        RegCloseKey(hkey);
    }
    return dpi;
}
524

525

526
/***********************************************************************
527
 *           GDI_inc_ref_count
528 529 530
 *
 * Increment the reference count of a GDI object.
 */
531
BOOL GDI_inc_ref_count( HGDIOBJ handle )
532
{
533 534 535 536 537 538 539 540
    GDIOBJHDR *header;

    if ((header = GDI_GetObjPtr( handle, MAGIC_DONTCARE )))
    {
        header->dwCount++;
        GDI_ReleaseObj( handle );
    }
    return header != NULL;
541 542 543 544
}


/***********************************************************************
545
 *           GDI_dec_ref_count
546 547 548
 *
 * Decrement the reference count of a GDI object.
 */
549
BOOL GDI_dec_ref_count( HGDIOBJ handle )
550 551 552 553 554 555 556 557 558 559 560 561
{
    GDIOBJHDR *header;

    if ((header = GDI_GetObjPtr( handle, MAGIC_DONTCARE )))
    {
        if (header->dwCount) header->dwCount--;
        if (header->dwCount != 0x80000000) GDI_ReleaseObj( handle );
        else
        {
            /* handle delayed DeleteObject*/
            header->dwCount = 0;
            GDI_ReleaseObj( handle );
562
            TRACE( "executing delayed DeleteObject for %p\n", handle );
563 564 565
            DeleteObject( handle );
        }
    }
566
    return header != NULL;
567 568 569
}


Alexandre Julliard's avatar
Alexandre Julliard committed
570 571 572
/***********************************************************************
 *           GDI_Init
 *
Alexandre Julliard's avatar
Alexandre Julliard committed
573
 * GDI initialization.
Alexandre Julliard's avatar
Alexandre Julliard committed
574
 */
575
BOOL GDI_Init(void)
Alexandre Julliard's avatar
Alexandre Julliard committed
576
{
577
    LOGFONTW default_gui_font;
578
    const struct DefaultFontInfo* deffonts;
579 580
    int i;

581 582
    WineEngInit();

583 584 585 586 587 588 589 590 591 592 593 594 595 596
    /* create stock objects */
    stock_objects[WHITE_BRUSH]  = CreateBrushIndirect( &WhiteBrush );
    stock_objects[LTGRAY_BRUSH] = CreateBrushIndirect( &LtGrayBrush );
    stock_objects[GRAY_BRUSH]   = CreateBrushIndirect( &GrayBrush );
    stock_objects[DKGRAY_BRUSH] = CreateBrushIndirect( &DkGrayBrush );
    stock_objects[BLACK_BRUSH]  = CreateBrushIndirect( &BlackBrush );
    stock_objects[NULL_BRUSH]   = CreateBrushIndirect( &NullBrush );

    stock_objects[WHITE_PEN]    = CreatePenIndirect( &WhitePen );
    stock_objects[BLACK_PEN]    = CreatePenIndirect( &BlackPen );
    stock_objects[NULL_PEN]     = CreatePenIndirect( &NullPen );

    stock_objects[DEFAULT_PALETTE] = PALETTE_Init();
    stock_objects[DEFAULT_BITMAP]  = CreateBitmap( 1, 1, 1, 1, NULL );
Alexandre Julliard's avatar
Alexandre Julliard committed
597

598
    /* language-independent stock fonts */
599 600 601
    stock_objects[OEM_FIXED_FONT]      = CreateFontIndirectW( &OEMFixedFont );
    stock_objects[ANSI_FIXED_FONT]     = CreateFontIndirectW( &AnsiFixedFont );
    stock_objects[ANSI_VAR_FONT]       = CreateFontIndirectW( &AnsiVarFont );
602 603 604

    /* language-dependent stock fonts */
    deffonts = get_default_fonts(get_default_charset());
605 606 607
    stock_objects[SYSTEM_FONT]         = CreateFontIndirectW( &deffonts->SystemFont );
    stock_objects[DEVICE_DEFAULT_FONT] = CreateFontIndirectW( &deffonts->DeviceDefaultFont );
    stock_objects[SYSTEM_FIXED_FONT]   = CreateFontIndirectW( &deffonts->SystemFixedFont );
608 609 610 611 612

    /* For the default gui font, we use the lfHeight member in deffonts as a place-holder
       for the point size so we must convert this into a true height */
    memcpy(&default_gui_font, &deffonts->DefaultGuiFont, sizeof(default_gui_font));
    default_gui_font.lfHeight = -MulDiv(default_gui_font.lfHeight, get_dpi(), 72);
613
    stock_objects[DEFAULT_GUI_FONT]    = CreateFontIndirectW( &default_gui_font );
Alexandre Julliard's avatar
Alexandre Julliard committed
614

615 616
    stock_objects[DC_BRUSH]     = CreateBrushIndirect( &DCBrush );
    stock_objects[DC_PEN]       = CreatePenIndirect( &DCPen );
Alexandre Julliard's avatar
Alexandre Julliard committed
617

618 619 620 621 622 623 624 625 626
    /* clear the NOSYSTEM bit on all stock objects*/
    for (i = 0; i < NB_STOCK_OBJECTS; i++)
    {
        if (!stock_objects[i])
        {
            if (i == 9) continue;  /* there's no stock object 9 */
            ERR( "could not create stock object %d\n", i );
            return FALSE;
        }
627
        __wine_make_gdi_object_system( stock_objects[i], TRUE );
628 629
    }

Alexandre Julliard's avatar
Alexandre Julliard committed
630
    return TRUE;
Alexandre Julliard's avatar
Alexandre Julliard committed
631 632
}

633
#define FIRST_LARGE_HANDLE 16
634
#define MAX_LARGE_HANDLES ((GDI_HEAP_SIZE >> 2) - FIRST_LARGE_HANDLE)
635 636 637 638 639 640 641 642
static GDIOBJHDR *large_handles[MAX_LARGE_HANDLES];
static int next_large_handle;

/***********************************************************************
 *           alloc_large_heap
 *
 * Allocate a GDI handle from the large heap. Helper for GDI_AllocObject
 */
643
static inline GDIOBJHDR *alloc_large_heap( WORD size, HGDIOBJ *handle )
644 645 646 647 648 649 650 651 652 653 654 655 656 657 658
{
    int i;
    GDIOBJHDR *obj;

    for (i = next_large_handle + 1; i < MAX_LARGE_HANDLES; i++)
        if (!large_handles[i]) goto found;
    for (i = 0; i <= next_large_handle; i++)
        if (!large_handles[i]) goto found;
    *handle = 0;
    return NULL;

 found:
    if ((obj = HeapAlloc( GetProcessHeap(), 0, size )))
    {
        large_handles[i] = obj;
659
        *handle = (HGDIOBJ)(ULONG_PTR)((i + FIRST_LARGE_HANDLE) << 2);
660 661 662 663 664
        next_large_handle = i;
    }
    return obj;
}

Alexandre Julliard's avatar
Alexandre Julliard committed
665 666 667 668

/***********************************************************************
 *           GDI_AllocObject
 */
669
void *GDI_AllocObject( WORD size, WORD magic, HGDIOBJ *handle, const struct gdi_obj_funcs *funcs )
Alexandre Julliard's avatar
Alexandre Julliard committed
670
{
671
    GDIOBJHDR *obj = NULL;
672 673

    _EnterSysLevel( &GDI_level );
674
    if (!(obj = alloc_large_heap( size, handle ))) goto error;
675

676
    obj->wMagic  = magic|OBJECT_NOSYSTEM;
677
    obj->dwCount = 0;
678
    obj->funcs   = funcs;
679
    obj->hdcs    = NULL;
680

681
    TRACE("(%p): enter %d\n", *handle, GDI_level.crst.RecursionCount);
682
    return obj;
683 684 685 686 687

error:
    _LeaveSysLevel( &GDI_level );
    *handle = 0;
    return NULL;
Alexandre Julliard's avatar
Alexandre Julliard committed
688 689 690 691
}


/***********************************************************************
692 693 694 695
 *           GDI_ReallocObject
 *
 * The object ptr must have been obtained with GDI_GetObjPtr.
 * The new pointer must be released with GDI_ReleaseObj.
Alexandre Julliard's avatar
Alexandre Julliard committed
696
 */
697
void *GDI_ReallocObject( WORD size, HGDIOBJ handle, void *object )
Alexandre Julliard's avatar
Alexandre Julliard committed
698
{
699
    void *new_ptr = NULL;
700
    int i;
Alexandre Julliard's avatar
Alexandre Julliard committed
701

702 703
    i = ((ULONG_PTR)handle >> 2) - FIRST_LARGE_HANDLE;
    if (i >= 0 && i < MAX_LARGE_HANDLES && large_handles[i])
704
    {
705 706
        new_ptr = HeapReAlloc( GetProcessHeap(), 0, large_handles[i], size );
        if (new_ptr) large_handles[i] = new_ptr;
707
    }
708
    else ERR( "Invalid handle %p\n", handle );
709 710
    if (!new_ptr)
    {
711
        TRACE("(%p): leave %d\n", handle, GDI_level.crst.RecursionCount);
712 713 714
        _LeaveSysLevel( &GDI_level );
    }
    return new_ptr;
715
}
716

717 718 719 720 721 722 723

/***********************************************************************
 *           GDI_FreeObject
 */
BOOL GDI_FreeObject( HGDIOBJ handle, void *ptr )
{
    GDIOBJHDR *object = ptr;
724
    int i;
725

726
    object->wMagic = 0;  /* Mark it as invalid */
727
    object->funcs  = NULL;
728
    i = ((ULONG_PTR)handle >> 2) - FIRST_LARGE_HANDLE;
729
    if (i >= 0 && i < MAX_LARGE_HANDLES)
730
    {
731 732
        HeapFree( GetProcessHeap(), 0, large_handles[i] );
        large_handles[i] = NULL;
733
    }
734
    else ERR( "Invalid handle %p\n", handle );
735
    TRACE("(%p): leave %d\n", handle, GDI_level.crst.RecursionCount);
736
    _LeaveSysLevel( &GDI_level );
Alexandre Julliard's avatar
Alexandre Julliard committed
737 738 739
    return TRUE;
}

740

Alexandre Julliard's avatar
Alexandre Julliard committed
741 742 743 744 745
/***********************************************************************
 *           GDI_GetObjPtr
 *
 * Return a pointer to the GDI object associated to the handle.
 * Return NULL if the object has the wrong magic number.
746
 * The object must be released with GDI_ReleaseObj.
Alexandre Julliard's avatar
Alexandre Julliard committed
747
 */
748
void *GDI_GetObjPtr( HGDIOBJ handle, WORD magic )
Alexandre Julliard's avatar
Alexandre Julliard committed
749
{
750
    GDIOBJHDR *ptr = NULL;
751
    int i;
752 753

    _EnterSysLevel( &GDI_level );
Alexandre Julliard's avatar
Alexandre Julliard committed
754

755 756
    i = ((UINT_PTR)handle >> 2) - FIRST_LARGE_HANDLE;
    if (i >= 0 && i < MAX_LARGE_HANDLES)
Alexandre Julliard's avatar
Alexandre Julliard committed
757
    {
758 759
        ptr = large_handles[i];
        if (ptr && (magic != MAGIC_DONTCARE) && (GDIMAGIC(ptr->wMagic) != magic)) ptr = NULL;
760
    }
761 762 763 764

    if (!ptr)
    {
        _LeaveSysLevel( &GDI_level );
765
        WARN( "Invalid handle %p\n", handle );
766
    }
767
    else TRACE("(%p): enter %d\n", handle, GDI_level.crst.RecursionCount);
768

Alexandre Julliard's avatar
Alexandre Julliard committed
769 770 771 772
    return ptr;
}


773 774 775 776 777 778
/***********************************************************************
 *           GDI_ReleaseObj
 *
 */
void GDI_ReleaseObj( HGDIOBJ handle )
{
779
    TRACE("(%p): leave %d\n", handle, GDI_level.crst.RecursionCount);
780 781 782 783
    _LeaveSysLevel( &GDI_level );
}


784 785 786 787 788 789 790 791 792
/***********************************************************************
 *           GDI_CheckNotLock
 */
void GDI_CheckNotLock(void)
{
    _CheckNotSysLevel( &GDI_level );
}


Alexandre Julliard's avatar
Alexandre Julliard committed
793
/***********************************************************************
794
 *           DeleteObject    (GDI32.@)
795 796 797 798 799 800 801 802 803 804 805
 *
 * Delete a Gdi object.
 *
 * PARAMS
 *  obj [I] Gdi object to delete
 *
 * RETURNS
 *  Success: TRUE. If obj was not returned from GetStockObject(), any resources
 *           it consumed are released.
 *  Failure: FALSE, if obj is not a valid Gdi object, or is currently selected
 *           into a DC.
Alexandre Julliard's avatar
Alexandre Julliard committed
806
 */
807
BOOL WINAPI DeleteObject( HGDIOBJ obj )
Alexandre Julliard's avatar
Alexandre Julliard committed
808 809 810
{
      /* Check if object is valid */

Alexandre Julliard's avatar
Alexandre Julliard committed
811 812
    GDIOBJHDR * header;
    if (HIWORD(obj)) return FALSE;
813

814
    if (!(header = GDI_GetObjPtr( obj, MAGIC_DONTCARE ))) return FALSE;
Alexandre Julliard's avatar
Alexandre Julliard committed
815

816 817 818
    if (!(header->wMagic & OBJECT_NOSYSTEM)
    &&   (header->wMagic >= FIRST_MAGIC) && (header->wMagic <= LAST_MAGIC))
    {
819
	TRACE("Preserving system object %p\n", obj);
820 821 822
        GDI_ReleaseObj( obj );
	return TRUE;
    }
823

824 825 826 827 828 829 830 831 832 833
    while (header->hdcs)
    {
        DC *dc = DC_GetDCPtr(header->hdcs->hdc);
        struct hdc_list *tmp;

        TRACE("hdc %p has interest in %p\n", header->hdcs->hdc, obj);
        if(dc)
        {
            if(dc->funcs->pDeleteObject)
                dc->funcs->pDeleteObject( dc->physDev, obj );
834
            DC_ReleaseDCPtr( dc );
835 836 837 838 839 840
        }
        tmp = header->hdcs;
        header->hdcs = header->hdcs->next;
        HeapFree(GetProcessHeap(), 0, tmp);
    }

841 842
    if (header->dwCount)
    {
843
        TRACE("delayed for %p because object in use, count %d\n", obj, header->dwCount );
844 845 846 847 848
        header->dwCount |= 0x80000000; /* mark for delete */
        GDI_ReleaseObj( obj );
        return TRUE;
    }

849
    TRACE("%p\n", obj );
Alexandre Julliard's avatar
Alexandre Julliard committed
850 851 852

      /* Delete object */

853 854
    if (header->funcs && header->funcs->pDeleteObject)
        return header->funcs->pDeleteObject( obj, header );
855

856
    GDI_ReleaseObj( obj );
Alexandre Julliard's avatar
Alexandre Julliard committed
857 858 859
    return FALSE;
}

860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931
/***********************************************************************
 *           GDI_hdc_using_object
 *
 * Call this if the dc requires DeleteObject notification
 */
BOOL GDI_hdc_using_object(HGDIOBJ obj, HDC hdc)
{
    GDIOBJHDR * header;
    struct hdc_list **pphdc;

    TRACE("obj %p hdc %p\n", obj, hdc);

    if (!(header = GDI_GetObjPtr( obj, MAGIC_DONTCARE ))) return FALSE;

    if (!(header->wMagic & OBJECT_NOSYSTEM) &&
         (header->wMagic >= FIRST_MAGIC) && (header->wMagic <= LAST_MAGIC))
    {
        GDI_ReleaseObj(obj);
        return FALSE;
    }

    for(pphdc = &header->hdcs; *pphdc; pphdc = &(*pphdc)->next)
        if((*pphdc)->hdc == hdc)
            break;

    if(!*pphdc) {
        *pphdc = HeapAlloc(GetProcessHeap(), 0, sizeof(**pphdc));
        (*pphdc)->hdc = hdc;
        (*pphdc)->next = NULL;
    }

    GDI_ReleaseObj(obj);
    return TRUE;
}

/***********************************************************************
 *           GDI_hdc_not_using_object
 *
 */
BOOL GDI_hdc_not_using_object(HGDIOBJ obj, HDC hdc)
{
    GDIOBJHDR * header;
    struct hdc_list *phdc, **prev;

    TRACE("obj %p hdc %p\n", obj, hdc);

    if (!(header = GDI_GetObjPtr( obj, MAGIC_DONTCARE ))) return FALSE;

    if (!(header->wMagic & OBJECT_NOSYSTEM) &&
         (header->wMagic >= FIRST_MAGIC) && (header->wMagic <= LAST_MAGIC))
    {
        GDI_ReleaseObj(obj);
        return FALSE;
    }

    phdc = header->hdcs;
    prev = &header->hdcs;

    while(phdc) {
        if(phdc->hdc == hdc) {
            *prev = phdc->next;
            HeapFree(GetProcessHeap(), 0, phdc);
            phdc = *prev;
        } else {
            prev = &phdc->next;
            phdc = phdc->next;
        }
    }

    GDI_ReleaseObj(obj);
    return TRUE;
}
Alexandre Julliard's avatar
Alexandre Julliard committed
932 933

/***********************************************************************
934
 *           GetStockObject    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
935
 */
936
HGDIOBJ WINAPI GetStockObject( INT obj )
Alexandre Julliard's avatar
Alexandre Julliard committed
937
{
938
    HGDIOBJ ret;
Alexandre Julliard's avatar
Alexandre Julliard committed
939
    if ((obj < 0) || (obj >= NB_STOCK_OBJECTS)) return 0;
940
    ret = stock_objects[obj];
941
    TRACE("returning %p\n", ret );
942
    return ret;
Alexandre Julliard's avatar
Alexandre Julliard committed
943 944 945 946
}


/***********************************************************************
947
 *           GetObject    (GDI.82)
Alexandre Julliard's avatar
Alexandre Julliard committed
948
 */
949
INT16 WINAPI GetObject16( HGDIOBJ16 handle16, INT16 count, LPVOID buffer )
Alexandre Julliard's avatar
Alexandre Julliard committed
950
{
951
    GDIOBJHDR * ptr;
952
    HGDIOBJ handle = HGDIOBJ_32( handle16 );
Alexandre Julliard's avatar
Alexandre Julliard committed
953
    INT16 result = 0;
954

955
    TRACE("%p %d %p\n", handle, count, buffer );
Alexandre Julliard's avatar
Alexandre Julliard committed
956 957
    if (!count) return 0;

958
    if (!(ptr = GDI_GetObjPtr( handle, MAGIC_DONTCARE ))) return 0;
959 960 961 962 963 964

    if (ptr->funcs && ptr->funcs->pGetObject16)
        result = ptr->funcs->pGetObject16( handle, ptr, count, buffer );
    else
        SetLastError( ERROR_INVALID_HANDLE );

965
    GDI_ReleaseObj( handle );
Alexandre Julliard's avatar
Alexandre Julliard committed
966
    return result;
Alexandre Julliard's avatar
Alexandre Julliard committed
967 968 969
}


Alexandre Julliard's avatar
Alexandre Julliard committed
970
/***********************************************************************
971
 *           GetObjectA    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
972
 */
973
INT WINAPI GetObjectA( HGDIOBJ handle, INT count, LPVOID buffer )
Alexandre Julliard's avatar
Alexandre Julliard committed
974
{
975
    GDIOBJHDR * ptr;
976
    INT result = 0;
977
    TRACE("%p %d %p\n", handle, count, buffer );
Alexandre Julliard's avatar
Alexandre Julliard committed
978

979 980
    if (!(ptr = GDI_GetObjPtr( handle, MAGIC_DONTCARE ))) return 0;

981 982 983 984
    if (ptr->funcs && ptr->funcs->pGetObjectA)
        result = ptr->funcs->pGetObjectA( handle, ptr, count, buffer );
    else
        SetLastError( ERROR_INVALID_HANDLE );
985

986
    GDI_ReleaseObj( handle );
Alexandre Julliard's avatar
Alexandre Julliard committed
987
    return result;
Alexandre Julliard's avatar
Alexandre Julliard committed
988
}
989

990
/***********************************************************************
991
 *           GetObjectW    (GDI32.@)
992
 */
993
INT WINAPI GetObjectW( HGDIOBJ handle, INT count, LPVOID buffer )
994
{
995
    GDIOBJHDR * ptr;
996
    INT result = 0;
997
    TRACE("%p %d %p\n", handle, count, buffer );
998

999 1000
    if (!(ptr = GDI_GetObjPtr( handle, MAGIC_DONTCARE ))) return 0;

1001 1002 1003 1004 1005
    if (ptr->funcs && ptr->funcs->pGetObjectW)
        result = ptr->funcs->pGetObjectW( handle, ptr, count, buffer );
    else
        SetLastError( ERROR_INVALID_HANDLE );

1006
    GDI_ReleaseObj( handle );
1007 1008
    return result;
}
Alexandre Julliard's avatar
Alexandre Julliard committed
1009

Alexandre Julliard's avatar
Alexandre Julliard committed
1010
/***********************************************************************
1011
 *           GetObjectType    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1012
 */
1013
DWORD WINAPI GetObjectType( HGDIOBJ handle )
Alexandre Julliard's avatar
Alexandre Julliard committed
1014
{
1015
    GDIOBJHDR * ptr;
1016
    INT result = 0;
1017
    TRACE("%p\n", handle );
Alexandre Julliard's avatar
Alexandre Julliard committed
1018

1019 1020 1021 1022 1023
    if (!(ptr = GDI_GetObjPtr( handle, MAGIC_DONTCARE )))
    {
        SetLastError( ERROR_INVALID_HANDLE );
        return 0;
    }
1024

1025
    switch(GDIMAGIC(ptr->wMagic))
Alexandre Julliard's avatar
Alexandre Julliard committed
1026 1027 1028 1029
    {
      case PEN_MAGIC:
	  result = OBJ_PEN;
	  break;
1030 1031 1032
      case EXT_PEN_MAGIC:
	  result = OBJ_EXTPEN;
	  break;
1033
      case BRUSH_MAGIC:
Alexandre Julliard's avatar
Alexandre Julliard committed
1034 1035
	  result = OBJ_BRUSH;
	  break;
1036
      case BITMAP_MAGIC:
Alexandre Julliard's avatar
Alexandre Julliard committed
1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059
	  result = OBJ_BITMAP;
	  break;
      case FONT_MAGIC:
	  result = OBJ_FONT;
	  break;
      case PALETTE_MAGIC:
	  result = OBJ_PAL;
	  break;
      case REGION_MAGIC:
	  result = OBJ_REGION;
	  break;
      case DC_MAGIC:
	  result = OBJ_DC;
	  break;
      case META_DC_MAGIC:
	  result = OBJ_METADC;
	  break;
      case METAFILE_MAGIC:
	  result = OBJ_METAFILE;
	  break;
      case METAFILE_DC_MAGIC:
	  result = OBJ_METADC;
	  break;
1060 1061 1062 1063 1064 1065
      case ENHMETAFILE_MAGIC:
	  result = OBJ_ENHMETAFILE;
	  break;
      case ENHMETAFILE_DC_MAGIC:
	  result = OBJ_ENHMETADC;
	  break;
1066 1067 1068
      case MEMORY_DC_MAGIC:
	  result = OBJ_MEMDC;
	  break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1069
      default:
1070
	  FIXME("Magic %04x not implemented\n", GDIMAGIC(ptr->wMagic) );
Alexandre Julliard's avatar
Alexandre Julliard committed
1071 1072
	  break;
    }
1073
    GDI_ReleaseObj( handle );
Alexandre Julliard's avatar
Alexandre Julliard committed
1074 1075
    return result;
}
Alexandre Julliard's avatar
Alexandre Julliard committed
1076

Alexandre Julliard's avatar
Alexandre Julliard committed
1077
/***********************************************************************
1078
 *           GetCurrentObject    	(GDI32.@)
1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096
 *
 * Get the currently selected object of a given type in a device context.
 *
 * PARAMS
 *  hdc  [I] Device context to get the current object from
 *  type [I] Type of current object to get (OBJ_* defines from "wingdi.h")
 *
 * RETURNS
 *  Success: The current object of the given type selected in hdc.
 *  Failure: A NULL handle.
 *
 * NOTES
 * - only the following object types are supported:
 *| OBJ_PEN
 *| OBJ_BRUSH
 *| OBJ_PAL
 *| OBJ_FONT
 *| OBJ_BITMAP
Alexandre Julliard's avatar
Alexandre Julliard committed
1097
 */
1098
HGDIOBJ WINAPI GetCurrentObject(HDC hdc,UINT type)
Alexandre Julliard's avatar
Alexandre Julliard committed
1099
{
1100
    HGDIOBJ ret = 0;
Alexandre Julliard's avatar
Alexandre Julliard committed
1101 1102
    DC * dc = DC_GetDCPtr( hdc );

1103
    if (dc)
1104
    {
Alexandre Julliard's avatar
Alexandre Julliard committed
1105
    switch (type) {
1106
	case OBJ_EXTPEN: /* fall through */
1107 1108 1109 1110 1111
	case OBJ_PEN:	 ret = dc->hPen; break;
	case OBJ_BRUSH:	 ret = dc->hBrush; break;
	case OBJ_PAL:	 ret = dc->hPalette; break;
	case OBJ_FONT:	 ret = dc->hFont; break;
	case OBJ_BITMAP: ret = dc->hBitmap; break;
1112 1113 1114

	/* tests show that OBJ_REGION is explicitly ignored */
	case OBJ_REGION: break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1115 1116
    default:
    	/* the SDK only mentions those above */
1117
    	FIXME("(%p,%d): unknown type.\n",hdc,type);
1118 1119
	    break;
        }
1120
        DC_ReleaseDCPtr( dc );
Alexandre Julliard's avatar
Alexandre Julliard committed
1121
    }
1122
    return ret;
Alexandre Julliard's avatar
Alexandre Julliard committed
1123
}
1124 1125


Alexandre Julliard's avatar
Alexandre Julliard committed
1126
/***********************************************************************
1127
 *           SelectObject    (GDI32.@)
1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140
 *
 * Select a Gdi object into a device context.
 *
 * PARAMS
 *  hdc  [I] Device context to associate the object with
 *  hObj [I] Gdi object to associate with hdc
 *
 * RETURNS
 *  Success: A non-NULL handle representing the previously selected object of
 *           the same type as hObj.
 *  Failure: A NULL object. If hdc is invalid, GetLastError() returns ERROR_INVALID_HANDLE.
 *           if hObj is not a valid object handle, no last error is set. In either
 *           case, hdc is unaffected by the call.
Alexandre Julliard's avatar
Alexandre Julliard committed
1141
 */
1142
HGDIOBJ WINAPI SelectObject( HDC hdc, HGDIOBJ hObj )
Alexandre Julliard's avatar
Alexandre Julliard committed
1143
{
1144
    HGDIOBJ ret = 0;
1145 1146
    GDIOBJHDR *header;
    DC *dc;
1147

1148
    TRACE( "(%p,%p)\n", hdc, hObj );
1149

1150 1151 1152
    if (!(dc = DC_GetDCPtr( hdc )))
        SetLastError( ERROR_INVALID_HANDLE );
    else
1153
    {
1154
        DC_ReleaseDCPtr( dc );
1155 1156 1157

        header = GDI_GetObjPtr( hObj, MAGIC_DONTCARE );
        if (header)
1158
        {
1159 1160 1161
            const struct gdi_obj_funcs *funcs = header->funcs;
            GDI_ReleaseObj( hObj );
            if (funcs && funcs->pSelectObject) ret = funcs->pSelectObject( hObj, hdc );
1162
        }
1163
    }
1164
    return ret;
Alexandre Julliard's avatar
Alexandre Julliard committed
1165 1166 1167
}


Alexandre Julliard's avatar
Alexandre Julliard committed
1168
/***********************************************************************
1169
 *           UnrealizeObject    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1170
 */
1171
BOOL WINAPI UnrealizeObject( HGDIOBJ obj )
Alexandre Julliard's avatar
Alexandre Julliard committed
1172
{
1173
    BOOL result = TRUE;
Alexandre Julliard's avatar
Alexandre Julliard committed
1174
  /* Check if object is valid */
Alexandre Julliard's avatar
Alexandre Julliard committed
1175

1176
    GDIOBJHDR * header = GDI_GetObjPtr( obj, MAGIC_DONTCARE );
Alexandre Julliard's avatar
Alexandre Julliard committed
1177 1178
    if (!header) return FALSE;

1179
    TRACE("%p\n", obj );
Alexandre Julliard's avatar
Alexandre Julliard committed
1180 1181 1182

      /* Unrealize object */

1183 1184
    if (header->funcs && header->funcs->pUnrealizeObject)
        result = header->funcs->pUnrealizeObject( obj, header );
Alexandre Julliard's avatar
Alexandre Julliard committed
1185

1186
    GDI_ReleaseObj( obj );
Alexandre Julliard's avatar
Alexandre Julliard committed
1187
    return result;
Alexandre Julliard's avatar
Alexandre Julliard committed
1188
}
Alexandre Julliard's avatar
Alexandre Julliard committed
1189 1190


Andreas Mohr's avatar
Andreas Mohr committed
1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201
/* 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)
};
1202

Alexandre Julliard's avatar
Alexandre Julliard committed
1203 1204

/***********************************************************************
1205
 *           EnumObjects    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1206
 */
1207 1208
INT WINAPI EnumObjects( HDC hdc, INT nObjType,
                            GOBJENUMPROC lpEnumFunc, LPARAM lParam )
Alexandre Julliard's avatar
Alexandre Julliard committed
1209
{
1210 1211
    UINT i;
    INT retval = 0;
1212 1213
    LOGPEN pen;
    LOGBRUSH brush;
Alexandre Julliard's avatar
Alexandre Julliard committed
1214

1215
    TRACE("%p %d %p %08lx\n", hdc, nObjType, lpEnumFunc, lParam );
Alexandre Julliard's avatar
Alexandre Julliard committed
1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226
    switch(nObjType)
    {
    case OBJ_PEN:
        /* Enumerate solid pens */
        for (i = 0; i < sizeof(solid_colors)/sizeof(solid_colors[0]); i++)
        {
            pen.lopnStyle   = PS_SOLID;
            pen.lopnWidth.x = 1;
            pen.lopnWidth.y = 0;
            pen.lopnColor   = solid_colors[i];
            retval = lpEnumFunc( &pen, lParam );
1227
            TRACE("solid pen %08x, ret=%d\n",
Alexandre Julliard's avatar
Alexandre Julliard committed
1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240
                         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++)
        {
            brush.lbStyle = BS_SOLID;
            brush.lbColor = solid_colors[i];
            brush.lbHatch = 0;
            retval = lpEnumFunc( &brush, lParam );
1241
            TRACE("solid brush %08x, ret=%d\n",
Alexandre Julliard's avatar
Alexandre Julliard committed
1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252
                         solid_colors[i], retval);
            if (!retval) break;
        }

        /* Now enumerate hatched brushes */
        if (retval) for (i = HS_HORIZONTAL; i <= HS_DIAGCROSS; i++)
        {
            brush.lbStyle = BS_HATCHED;
            brush.lbColor = RGB(0,0,0);
            brush.lbHatch = i;
            retval = lpEnumFunc( &brush, lParam );
1253
            TRACE("hatched brush %d, ret=%d\n",
Alexandre Julliard's avatar
Alexandre Julliard committed
1254 1255 1256 1257 1258 1259 1260
                         i, retval);
            if (!retval) break;
        }
        break;

    default:
        /* FIXME: implement Win32 types */
1261
        WARN("(%d): Invalid type\n", nObjType );
Alexandre Julliard's avatar
Alexandre Julliard committed
1262 1263 1264
        break;
    }
    return retval;
Alexandre Julliard's avatar
Alexandre Julliard committed
1265
}
Alexandre Julliard's avatar
Alexandre Julliard committed
1266

Alexandre Julliard's avatar
Alexandre Julliard committed
1267

Alexandre Julliard's avatar
Alexandre Julliard committed
1268
/***********************************************************************
Alexandre Julliard's avatar
Alexandre Julliard committed
1269
 *           IsGDIObject    (GDI.462)
1270
 *
Alexandre Julliard's avatar
Alexandre Julliard committed
1271
 * returns type of object if valid (W95 system programming secrets p. 264-5)
Alexandre Julliard's avatar
Alexandre Julliard committed
1272
 */
1273
BOOL16 WINAPI IsGDIObject16( HGDIOBJ16 handle16 )
Alexandre Julliard's avatar
Alexandre Julliard committed
1274
{
Alexandre Julliard's avatar
Alexandre Julliard committed
1275
    UINT16 magic = 0;
1276
    HGDIOBJ handle = HGDIOBJ_32( handle16 );
Alexandre Julliard's avatar
Alexandre Julliard committed
1277

1278 1279
    GDIOBJHDR *object = GDI_GetObjPtr( handle, MAGIC_DONTCARE );
    if (object)
Alexandre Julliard's avatar
Alexandre Julliard committed
1280
    {
1281
        magic = GDIMAGIC(object->wMagic) - FIRST_MAGIC + 1;
1282
        GDI_ReleaseObj( handle );
Alexandre Julliard's avatar
Alexandre Julliard committed
1283
    }
1284
    return magic;
Alexandre Julliard's avatar
Alexandre Julliard committed
1285
}
Alexandre Julliard's avatar
Alexandre Julliard committed
1286 1287


Alexandre Julliard's avatar
Alexandre Julliard committed
1288
/***********************************************************************
1289
 *           SetObjectOwner    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1290
 */
1291
void WINAPI SetObjectOwner( HGDIOBJ handle, HANDLE owner )
Alexandre Julliard's avatar
Alexandre Julliard committed
1292 1293 1294 1295
{
    /* Nothing to do */
}

1296

Alexandre Julliard's avatar
Alexandre Julliard committed
1297 1298
/***********************************************************************
 *           MakeObjectPrivate    (GDI.463)
1299 1300 1301 1302 1303 1304 1305
 *
 * What does that mean ?
 * Some little docu can be found in "Undocumented Windows",
 * but this is basically useless.
 * At least we know that this flags the GDI object's wMagic
 * with 0x2000 (OBJECT_PRIVATE), so we just do it.
 * But Wine doesn't react on that yet.
Alexandre Julliard's avatar
Alexandre Julliard committed
1306
 */
1307
void WINAPI MakeObjectPrivate16( HGDIOBJ16 handle16, BOOL16 private )
Alexandre Julliard's avatar
Alexandre Julliard committed
1308
{
1309
    HGDIOBJ handle = HGDIOBJ_32( handle16 );
1310 1311 1312
    GDIOBJHDR *ptr = GDI_GetObjPtr( handle, MAGIC_DONTCARE );
    if (!ptr)
    {
1313
	ERR("invalid GDI object %p !\n", handle);
1314 1315 1316 1317
	return;
    }
    ptr->wMagic |= OBJECT_PRIVATE;
    GDI_ReleaseObj( handle );
Alexandre Julliard's avatar
Alexandre Julliard committed
1318 1319 1320
}


Alexandre Julliard's avatar
Alexandre Julliard committed
1321
/***********************************************************************
1322
 *           GdiFlush    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1323
 */
1324
BOOL WINAPI GdiFlush(void)
Alexandre Julliard's avatar
Alexandre Julliard committed
1325 1326 1327 1328
{
    return TRUE;  /* FIXME */
}

Alexandre Julliard's avatar
Alexandre Julliard committed
1329 1330

/***********************************************************************
1331
 *           GdiGetBatchLimit    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1332 1333 1334 1335 1336 1337 1338 1339
 */
DWORD WINAPI GdiGetBatchLimit(void)
{
    return 1;  /* FIXME */
}


/***********************************************************************
1340
 *           GdiSetBatchLimit    (GDI32.@)
Alexandre Julliard's avatar
Alexandre Julliard committed
1341 1342 1343 1344 1345 1346 1347
 */
DWORD WINAPI GdiSetBatchLimit( DWORD limit )
{
    return 1; /* FIXME */
}


Alexandre Julliard's avatar
Alexandre Julliard committed
1348 1349 1350
/***********************************************************************
 *           GdiSeeGdiDo   (GDI.452)
 */
1351
DWORD WINAPI GdiSeeGdiDo16( WORD wReqType, WORD wParam1, WORD wParam2,
Alexandre Julliard's avatar
Alexandre Julliard committed
1352 1353
                          WORD wParam3 )
{
1354
    DWORD ret = ~0U;
1355

Alexandre Julliard's avatar
Alexandre Julliard committed
1356 1357 1358
    switch (wReqType)
    {
    case 0x0001:  /* LocalAlloc */
1359 1360
        WARN("LocalAlloc16(%x, %x): ignoring\n", wParam1, wParam3);
        ret = 0;
1361
        break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1362
    case 0x0002:  /* LocalFree */
1363 1364
        WARN("LocalFree16(%x): ignoring\n", wParam1);
        ret = 0;
1365
        break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1366
    case 0x0003:  /* LocalCompact */
1367 1368
        WARN("LocalCompact16(%x): ignoring\n", wParam3);
        ret = 65000; /* lie about the amount of free space */
1369
        break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1370
    case 0x0103:  /* LocalHeap */
1371
        WARN("LocalHeap16(): ignoring\n");
1372
        break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1373
    default:
1374
        WARN("(wReqType=%04x): Unknown\n", wReqType);
1375
        break;
Alexandre Julliard's avatar
Alexandre Julliard committed
1376
    }
1377
    return ret;
Alexandre Julliard's avatar
Alexandre Julliard committed
1378 1379
}

1380
/***********************************************************************
Patrik Stridvall's avatar
Patrik Stridvall committed
1381
 *           GdiSignalProc32     (GDI.610)
1382 1383 1384 1385 1386 1387 1388
 */
WORD WINAPI GdiSignalProc( UINT uCode, DWORD dwThreadOrProcessID,
                           DWORD dwFlags, HMODULE16 hModule )
{
    return 0;
}

1389
/***********************************************************************
1390
 *           GdiInit2     (GDI.403)
1391 1392
 *
 * See "Undocumented Windows"
1393 1394 1395 1396
 *
 * PARAMS
 *   h1 [I] GDI object
 *   h2 [I] global data
1397
 */
1398
HANDLE16 WINAPI GdiInit216( HANDLE16 h1, HANDLE16 h2 )
1399 1400 1401 1402 1403 1404 1405
{
    FIXME("(%04x, %04x), stub.\n", h1, h2);
    if (h2 == 0xffff)
	return 0xffff; /* undefined return value */
    return h1; /* FIXME: should be the memory handle of h1 */
}

1406
/***********************************************************************
1407
 *           FinalGdiInit     (GDI.405)
1408
 */
1409
void WINAPI FinalGdiInit16( HBRUSH16 hPattern /* [in] fill pattern of desktop */ )
1410 1411
{
}
1412

1413 1414 1415
/***********************************************************************
 *           GdiFreeResources   (GDI.609)
 */
1416
WORD WINAPI GdiFreeResources16( DWORD reserve )
1417
{
1418
    return 90; /* lie about it, it shouldn't matter */
1419 1420
}

Alexandre Julliard's avatar
Alexandre Julliard committed
1421

1422
/*******************************************************************
1423
 *      GetColorAdjustment [GDI32.@]
1424 1425 1426
 *
 *
 */
1427
BOOL WINAPI GetColorAdjustment(HDC hdc, LPCOLORADJUSTMENT lpca)
1428
{
1429
        FIXME("GetColorAdjustment, stub\n");
1430 1431 1432 1433
        return 0;
}

/*******************************************************************
1434
 *      GdiComment [GDI32.@]
1435 1436 1437
 *
 *
 */
1438
BOOL WINAPI GdiComment(HDC hdc, UINT cbSize, const BYTE *lpData)
1439
{
1440 1441 1442 1443 1444 1445
    DC *dc = DC_GetDCPtr(hdc);
    BOOL ret = FALSE;
    if(dc)
    {
        if (dc->funcs->pGdiComment)
            ret = dc->funcs->pGdiComment( dc->physDev, cbSize, lpData );
1446
        DC_ReleaseDCPtr( dc );
1447 1448
    }
    return ret;
1449
}
1450

1451
/*******************************************************************
1452
 *      SetColorAdjustment [GDI32.@]
1453 1454 1455
 *
 *
 */
1456
BOOL WINAPI SetColorAdjustment(HDC hdc, const COLORADJUSTMENT* lpca)
1457
{
1458
        FIXME("SetColorAdjustment, stub\n");
1459 1460
        return 0;
}