1 /*
2 * PROJECT: ReactOS user32.dll
3 * COPYRIGHT: GPL - See COPYING in the top level directory
4 * FILE: win32ss/user/user32/windows/cursoricon.c
5 * PURPOSE: cursor and icons implementation
6 * PROGRAMMER: Jérôme Gardou (jerome.gardou@reactos.org)
7 */
8
9 #include <user32.h>
10
11 WINE_DEFAULT_DEBUG_CHANNEL(cursor);
12 WINE_DECLARE_DEBUG_CHANNEL(icon);
13 //WINE_DECLARE_DEBUG_CHANNEL(resource);
14
15 /* We only use Wide string functions */
16 #undef MAKEINTRESOURCE
17 #define MAKEINTRESOURCE MAKEINTRESOURCEW
18
19 /************* USER32 INTERNAL FUNCTIONS **********/
20
LoadSystemCursors(VOID)21 VOID LoadSystemCursors(VOID)
22 {
23 if (!gpsi->hIconSmWindows)
24 {
25 ERR("Loading System Cursors\n");
26 NtUserSetSystemCursor(LoadImageW( 0, IDC_ARROW, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_NORMAL);
27 NtUserSetSystemCursor(LoadImageW( 0, IDC_IBEAM, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_IBEAM);
28 NtUserSetSystemCursor(LoadImageW( 0, IDC_WAIT, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_WAIT);
29 NtUserSetSystemCursor(LoadImageW( 0, IDC_CROSS, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_CROSS);
30 NtUserSetSystemCursor(LoadImageW( 0, IDC_UPARROW, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_UP);
31 NtUserSetSystemCursor(LoadImageW( 0, IDC_ICON, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_ICON);
32 NtUserSetSystemCursor(LoadImageW( 0, IDC_SIZE, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_SIZE);
33 NtUserSetSystemCursor(LoadImageW( 0, IDC_SIZENWSE, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_SIZENWSE);
34 NtUserSetSystemCursor(LoadImageW( 0, IDC_SIZENESW, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_SIZENESW);
35 NtUserSetSystemCursor(LoadImageW( 0, IDC_SIZEWE, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_SIZEWE);
36 NtUserSetSystemCursor(LoadImageW( 0, IDC_SIZENS, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_SIZENS);
37 NtUserSetSystemCursor(LoadImageW( 0, IDC_SIZEALL, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_SIZEALL);
38 NtUserSetSystemCursor(LoadImageW( 0, IDC_NO, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_NO);
39 NtUserSetSystemCursor(LoadImageW( 0, IDC_HAND, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_HAND);
40 NtUserSetSystemCursor(LoadImageW( 0, IDC_APPSTARTING, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_APPSTARTING);
41 NtUserSetSystemCursor(LoadImageW( 0, IDC_HELP, IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE ), OCR_HELP);
42 }
43 }
44
45 /* This callback routine is called directly after switching to gui mode */
46 NTSTATUS
47 WINAPI
User32SetupDefaultCursors(PVOID Arguments,ULONG ArgumentLength)48 User32SetupDefaultCursors(PVOID Arguments,
49 ULONG ArgumentLength)
50 {
51 BOOL *DefaultCursor = (BOOL*)Arguments;
52 HCURSOR hCursor;
53
54 /* Load system cursors first */
55 LoadSystemCursors();
56
57 if(*DefaultCursor)
58 {
59 /* set default cursor */
60 hCursor = LoadCursorW(0, IDC_ARROW);
61 SetCursor(hCursor);
62 }
63 else
64 {
65 /* FIXME load system cursor scheme */
66 SetCursor(0);
67 hCursor = LoadCursorW(0, IDC_ARROW);
68 SetCursor(hCursor);
69 }
70
71 return(ZwCallbackReturn(&hCursor, sizeof(HCURSOR), STATUS_SUCCESS));
72 }
73
get_icon_size(HICON hIcon,SIZE * size)74 BOOL get_icon_size(HICON hIcon, SIZE *size)
75 {
76 return NtUserGetIconSize(hIcon, 0, &size->cx, &size->cy);
77 }
78
CursorIconToCursor(HICON hIcon,BOOL SemiTransparent)79 HCURSOR CursorIconToCursor(HICON hIcon, BOOL SemiTransparent)
80 {
81 UNIMPLEMENTED;
82 return NULL;
83 }
84
85 /************* IMPLEMENTATION HELPERS ******************/
86
87 static const WCHAR DISPLAYW[] = L"DISPLAY";
88
map_fileW(LPCWSTR name,LPDWORD filesize)89 static void *map_fileW( LPCWSTR name, LPDWORD filesize )
90 {
91 HANDLE hFile, hMapping;
92 LPVOID ptr = NULL;
93
94 hFile = CreateFileW( name, GENERIC_READ, FILE_SHARE_READ, NULL,
95 OPEN_EXISTING, FILE_FLAG_RANDOM_ACCESS, 0 );
96 if (hFile != INVALID_HANDLE_VALUE)
97 {
98 hMapping = CreateFileMappingW( hFile, NULL, PAGE_READONLY, 0, 0, NULL );
99 if (hMapping)
100 {
101 ptr = MapViewOfFile( hMapping, FILE_MAP_READ, 0, 0, 0 );
102 CloseHandle( hMapping );
103 if (filesize)
104 *filesize = GetFileSize( hFile, NULL );
105 }
106 CloseHandle( hFile );
107 }
108 return ptr;
109 }
110
get_dib_image_size(int width,int height,int depth)111 static int get_dib_image_size( int width, int height, int depth )
112 {
113 return (((width * depth + 31) / 8) & ~3) * abs( height );
114 }
115
is_dib_monochrome(const BITMAPINFO * info)116 static BOOL is_dib_monochrome( const BITMAPINFO* info )
117 {
118 if (info->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
119 {
120 const RGBTRIPLE *rgb = ((const BITMAPCOREINFO*)info)->bmciColors;
121
122 if (((const BITMAPCOREINFO*)info)->bmciHeader.bcBitCount != 1) return FALSE;
123
124 /* Check if the first color is black */
125 if ((rgb->rgbtRed == 0) && (rgb->rgbtGreen == 0) && (rgb->rgbtBlue == 0))
126 {
127 rgb++;
128
129 /* Check if the second color is white */
130 return ((rgb->rgbtRed == 0xff) && (rgb->rgbtGreen == 0xff)
131 && (rgb->rgbtBlue == 0xff));
132 }
133 else return FALSE;
134 }
135 else /* assume BITMAPINFOHEADER */
136 {
137 const RGBQUAD *rgb = info->bmiColors;
138
139 if (info->bmiHeader.biBitCount != 1) return FALSE;
140
141 /* Check if the first color is black */
142 if ((rgb->rgbRed == 0) && (rgb->rgbGreen == 0) &&
143 (rgb->rgbBlue == 0) && (rgb->rgbReserved == 0))
144 {
145 rgb++;
146
147 /* Check if the second color is white */
148 return ((rgb->rgbRed == 0xff) && (rgb->rgbGreen == 0xff)
149 && (rgb->rgbBlue == 0xff) && (rgb->rgbReserved == 0));
150 }
151 else return FALSE;
152 }
153 }
154
155 /* Return the size of the bitmap info structure including color table and
156 * the bytes required for 3 DWORDS if this is a BI_BITFIELDS bmp. */
bitmap_info_size(const BITMAPINFO * info,WORD coloruse)157 static int bitmap_info_size( const BITMAPINFO * info, WORD coloruse )
158 {
159 unsigned int colors, size, masks = 0;
160
161 if (info->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
162 {
163 const BITMAPCOREHEADER *core = (const BITMAPCOREHEADER *)info;
164 colors = (core->bcBitCount <= 8) ? 1 << core->bcBitCount : 0;
165 return sizeof(BITMAPCOREHEADER) + colors *
166 ((coloruse == DIB_RGB_COLORS) ? sizeof(RGBTRIPLE) : sizeof(WORD));
167 }
168 else /* assume BITMAPINFOHEADER */
169 {
170 colors = info->bmiHeader.biClrUsed;
171 if (colors > 256) /* buffer overflow otherwise */
172 colors = 256;
173 if (!colors && (info->bmiHeader.biBitCount <= 8))
174 colors = 1 << info->bmiHeader.biBitCount;
175 /* Account for BI_BITFIELDS in BITMAPINFOHEADER(v1-v3) bmp's. The
176 * 'max' selection using biSize below will exclude v4 & v5's. */
177 if (info->bmiHeader.biCompression == BI_BITFIELDS) masks = 3;
178 size = max( info->bmiHeader.biSize, sizeof(BITMAPINFOHEADER) + masks * sizeof(DWORD) );
179 /* Test for BI_BITFIELDS format and either 16 or 32 bpp.
180 * If so, account for the 3 DWORD masks (RGB Order).
181 * BITMAPCOREHEADER tested above has no 16 or 32 bpp types.
182 * See table "All of the possible pixel formats in a DIB"
183 * at https://en.wikipedia.org/wiki/BMP_file_format. */
184 if (info->bmiHeader.biSize >= sizeof(BITMAPV4HEADER) &&
185 info->bmiHeader.biCompression == BI_BITFIELDS &&
186 (info->bmiHeader.biBitCount == 16 || info->bmiHeader.biBitCount == 32))
187 {
188 size += 3 * sizeof(DWORD); // BI_BITFIELDS
189 }
190 return size + colors * ((coloruse == DIB_RGB_COLORS) ? sizeof(RGBQUAD) : sizeof(WORD));
191 }
192 }
193
DIB_GetBitmapInfo(const BITMAPINFOHEADER * header,LONG * width,LONG * height,WORD * bpp,DWORD * compr)194 static int DIB_GetBitmapInfo( const BITMAPINFOHEADER *header, LONG *width,
195 LONG *height, WORD *bpp, DWORD *compr )
196 {
197 #define CR 13
198 #define LF 10
199 #define EOFM 26 // DOS End Of File Marker
200 #define HighBitDetect 0x89 // Byte with high bit set to test if not 7-bit
201 /* wine's definition */
202 static const BYTE png_sig_pattern[] = { HighBitDetect, 'P', 'N', 'G', CR, LF, EOFM, LF };
203 if (header->biSize == sizeof(BITMAPCOREHEADER))
204 {
205 const BITMAPCOREHEADER *core = (const BITMAPCOREHEADER *)header;
206 *width = core->bcWidth;
207 *height = core->bcHeight;
208 *bpp = core->bcBitCount;
209 *compr = 0;
210 return 0;
211 }
212 else if (header->biSize == sizeof(BITMAPINFOHEADER) ||
213 header->biSize == sizeof(BITMAPV4HEADER) ||
214 header->biSize == sizeof(BITMAPV5HEADER))
215 {
216 *width = header->biWidth;
217 *height = header->biHeight;
218 *bpp = header->biBitCount;
219 *compr = header->biCompression;
220 return 1;
221 }
222 if (memcmp(&header->biSize, png_sig_pattern, sizeof(png_sig_pattern)) == 0)
223 {
224 ERR("Cannot yet display PNG icons\n");
225 /* for PNG format details see https://en.wikipedia.org/wiki/PNG */
226 }
227 else
228 {
229 ERR("Unknown/wrong size for header of 0x%x\n", header->biSize );
230 }
231 return -1;
232 }
233
234 /* copy an icon bitmap, even when it can't be selected into a DC */
235 /* helper for CreateIconIndirect */
stretch_blt_icon(HDC hdc_dst,int dst_width,int dst_height,HBITMAP src)236 static void stretch_blt_icon(HDC hdc_dst, int dst_width, int dst_height, HBITMAP src)
237 {
238 HDC hdc = CreateCompatibleDC( 0 );
239 BITMAP bm;
240 HBITMAP hbmpPrev;
241
242 GetObjectW(src, sizeof(bm), &bm);
243
244 hbmpPrev = SelectObject(hdc, src);
245
246 if (!hbmpPrev) /* do it the hard way */
247 {
248 BITMAPINFO *info;
249 void *bits;
250
251 if (!(info = HeapAlloc( GetProcessHeap(), 0, FIELD_OFFSET( BITMAPINFO, bmiColors[256] )))) return;
252 info->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
253 info->bmiHeader.biWidth = bm.bmWidth;
254 info->bmiHeader.biHeight = bm.bmHeight;
255 info->bmiHeader.biPlanes = GetDeviceCaps( hdc_dst, PLANES );
256 info->bmiHeader.biBitCount = GetDeviceCaps( hdc_dst, BITSPIXEL );
257 info->bmiHeader.biCompression = BI_RGB;
258 info->bmiHeader.biSizeImage = get_dib_image_size( bm.bmWidth, bm.bmHeight, info->bmiHeader.biBitCount );
259 info->bmiHeader.biXPelsPerMeter = 0;
260 info->bmiHeader.biYPelsPerMeter = 0;
261 info->bmiHeader.biClrUsed = 0;
262 info->bmiHeader.biClrImportant = 0;
263 bits = HeapAlloc( GetProcessHeap(), 0, info->bmiHeader.biSizeImage );
264 if (bits && GetDIBits( hdc, src, 0, bm.bmHeight, bits, info, DIB_RGB_COLORS ))
265 StretchDIBits( hdc_dst, 0, 0, dst_width, dst_height,
266 0, 0, bm.bmWidth, bm.bmHeight, bits, info, DIB_RGB_COLORS, SRCCOPY );
267
268 HeapFree( GetProcessHeap(), 0, bits );
269 HeapFree( GetProcessHeap(), 0, info );
270 }
271 else
272 {
273 StretchBlt( hdc_dst, 0, 0, dst_width, dst_height, hdc, 0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY );
274 SelectObject(hdc, hbmpPrev);
275 }
276
277 DeleteDC( hdc );
278 }
279
280 /***********************************************************************
281 * bmi_has_alpha
282 */
bmi_has_alpha(const BITMAPINFO * info,const void * bits)283 static BOOL bmi_has_alpha( const BITMAPINFO *info, const void *bits )
284 {
285 int i;
286 BOOL has_alpha = FALSE;
287 const unsigned char *ptr = bits;
288
289 if (info->bmiHeader.biBitCount != 32) return FALSE;
290 for (i = 0; i < info->bmiHeader.biWidth * abs(info->bmiHeader.biHeight); i++, ptr += 4)
291 if ((has_alpha = (ptr[3] != 0))) break;
292 return has_alpha;
293 }
294
295 /***********************************************************************
296 * create_alpha_bitmap
297 *
298 * Create the alpha bitmap for a 32-bpp icon that has an alpha channel.
299 */
300 static
301 HBITMAP
create_alpha_bitmap(_In_opt_ HBITMAP color,_In_opt_ BITMAPINFO * src_info,_In_opt_ const void * color_bits,_In_ LONG width,_In_ LONG height)302 create_alpha_bitmap(
303 _In_opt_ HBITMAP color,
304 _In_opt_ BITMAPINFO *src_info,
305 _In_opt_ const void *color_bits,
306 _In_ LONG width,
307 _In_ LONG height)
308 {
309 HBITMAP alpha = NULL, hbmpOld;
310 HDC hdc = NULL, hdcScreen;
311 unsigned char *ptr;
312 void *bits = NULL;
313 ULONG size;
314
315 hdcScreen = CreateDCW(DISPLAYW, NULL, NULL, NULL);
316 if (!hdcScreen)
317 return NULL;
318 hdc = CreateCompatibleDC(hdcScreen);
319 if (!hdc)
320 {
321 DeleteDC(hdcScreen);
322 return NULL;
323 }
324
325 if (color)
326 {
327 BITMAP bm;
328 BITMAPINFO *info = NULL;
329
330 TRACE("Creating alpha bitmap from existing bitmap.\n");
331
332 if (!GetObjectW( color, sizeof(bm), &bm ))
333 goto done;
334 if (bm.bmBitsPixel != 32)
335 goto done;
336
337 size = get_dib_image_size(bm.bmWidth, bm.bmHeight, 32);
338
339 info = HeapAlloc(GetProcessHeap(), 0, FIELD_OFFSET(BITMAPINFO, bmiColors[256]));
340 if(!info)
341 goto done;
342 info->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
343 info->bmiHeader.biWidth = bm.bmWidth;
344 info->bmiHeader.biHeight = -bm.bmHeight;
345 info->bmiHeader.biPlanes = 1;
346 info->bmiHeader.biBitCount = 32;
347 info->bmiHeader.biCompression = BI_RGB;
348 info->bmiHeader.biSizeImage = size;
349 info->bmiHeader.biXPelsPerMeter = 0;
350 info->bmiHeader.biYPelsPerMeter = 0;
351 info->bmiHeader.biClrUsed = 0;
352 info->bmiHeader.biClrImportant = 0;
353
354 bits = HeapAlloc(GetProcessHeap(), 0, size);
355 if(!bits)
356 {
357 HeapFree(GetProcessHeap(), 0, info);
358 goto done;
359 }
360 if(!GetDIBits( hdc, color, 0, bm.bmHeight, bits, info, DIB_RGB_COLORS ))
361 {
362 HeapFree(GetProcessHeap(), 0, info);
363 goto done;
364 }
365 if (!bmi_has_alpha( info, bits ))
366 {
367 HeapFree(GetProcessHeap(), 0, info);
368 goto done;
369 }
370
371 /* pre-multiply by alpha */
372 for (ptr = bits; ptr < ((BYTE*)bits + size); ptr += 4)
373 {
374 unsigned int alpha = ptr[3];
375 ptr[0] = (ptr[0] * alpha) / 255;
376 ptr[1] = (ptr[1] * alpha) / 255;
377 ptr[2] = (ptr[2] * alpha) / 255;
378 }
379
380 /* Directly create a 32-bits DDB (thanks to undocumented CreateDIBitmap flag). */
381 alpha = CreateDIBitmap(hdc, NULL, CBM_INIT | 2, bits, info, DIB_RGB_COLORS);
382
383 HeapFree(GetProcessHeap(), 0, info);
384 }
385 else
386 {
387 WORD bpp;
388 DWORD compr;
389 LONG orig_width, orig_height;
390
391 TRACE("Creating alpha bitmap from bitmap info.\n");
392
393 if(!bmi_has_alpha(src_info, color_bits))
394 goto done;
395
396 if(!DIB_GetBitmapInfo(&src_info->bmiHeader, &orig_width, &orig_height, &bpp, &compr))
397 goto done;
398 if(bpp != 32)
399 goto done;
400
401 size = get_dib_image_size(orig_width, orig_height, bpp);
402 bits = HeapAlloc(GetProcessHeap(), 0, size);
403 if(!bits)
404 goto done;
405 CopyMemory(bits, color_bits, size);
406 /* pre-multiply by alpha */
407 for (ptr = bits; ptr < ((BYTE*)bits + size); ptr += 4)
408 {
409 unsigned int alpha = ptr[3];
410 ptr[0] = (ptr[0] * alpha) / 255;
411 ptr[1] = (ptr[1] * alpha) / 255;
412 ptr[2] = (ptr[2] * alpha) / 255;
413 }
414
415 /* Create the bitmap. Set the bitmap info to have the right width and height */
416 if(src_info->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
417 {
418 ((BITMAPCOREHEADER*)&src_info->bmiHeader)->bcWidth = width;
419 ((BITMAPCOREHEADER*)&src_info->bmiHeader)->bcHeight = height;
420 }
421 else
422 {
423 src_info->bmiHeader.biWidth = width;
424 src_info->bmiHeader.biHeight = height;
425 }
426 /* Directly create a 32-bits DDB (thanks to undocumented CreateDIBitmap flag). */
427 alpha = CreateDIBitmap(hdcScreen, NULL, 2, NULL, src_info, DIB_RGB_COLORS);
428 /* Restore values */
429 if(src_info->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
430 {
431 ((BITMAPCOREHEADER*)&src_info->bmiHeader)->bcWidth = orig_width;
432 ((BITMAPCOREHEADER*)&src_info->bmiHeader)->bcHeight = orig_height;
433 }
434 else
435 {
436 src_info->bmiHeader.biWidth = orig_width;
437 src_info->bmiHeader.biHeight = orig_height;
438 }
439 if(!alpha)
440 goto done;
441 hbmpOld = SelectObject(hdc, alpha);
442 if(!hbmpOld)
443 {
444 DeleteObject(alpha);
445 alpha = NULL;
446 goto done;
447 }
448 if(!StretchDIBits( hdc, 0, 0, width, height,
449 0, 0, orig_width, orig_height,
450 bits, src_info, DIB_RGB_COLORS, SRCCOPY ))
451 {
452 SelectObject(hdc, hbmpOld);
453 hbmpOld = NULL;
454 DeleteObject(alpha);
455 alpha = NULL;
456 }
457 else
458 {
459 SelectObject(hdc, hbmpOld);
460 }
461 }
462
463 done:
464 DeleteDC(hdcScreen);
465 DeleteDC( hdc );
466 if(bits) HeapFree(GetProcessHeap(), 0, bits);
467
468 TRACE("Returning 0x%08x.\n", alpha);
469 return alpha;
470 }
471
472 #include "pshpack1.h"
473
474 typedef struct {
475 BYTE bWidth;
476 BYTE bHeight;
477 BYTE bColorCount;
478 BYTE bReserved;
479 WORD xHotspot;
480 WORD yHotspot;
481 DWORD dwDIBSize;
482 DWORD dwDIBOffset;
483 } CURSORICONFILEDIRENTRY;
484
485 typedef struct
486 {
487 WORD idReserved;
488 WORD idType;
489 WORD idCount;
490 CURSORICONFILEDIRENTRY idEntries[1];
491 } CURSORICONFILEDIR;
492
493 #include "poppack.h"
494
495 const CURSORICONFILEDIRENTRY*
get_best_icon_file_entry(_In_ const CURSORICONFILEDIR * dir,_In_ DWORD dwFileSize,_In_ int cxDesired,_In_ int cyDesired,_In_ BOOL bIcon,_In_ DWORD fuLoad)496 get_best_icon_file_entry(
497 _In_ const CURSORICONFILEDIR* dir,
498 _In_ DWORD dwFileSize,
499 _In_ int cxDesired,
500 _In_ int cyDesired,
501 _In_ BOOL bIcon,
502 _In_ DWORD fuLoad
503 )
504 {
505 CURSORICONDIR* fakeDir;
506 CURSORICONDIRENTRY* fakeEntry;
507 WORD i;
508 const CURSORICONFILEDIRENTRY* entry;
509
510 /* Check our file is what it claims to be */
511 if ( dwFileSize < sizeof(*dir) )
512 return NULL;
513
514 if (dwFileSize < FIELD_OFFSET(CURSORICONFILEDIR, idEntries[dir->idCount]))
515 return NULL;
516
517 /*
518 * Cute little hack:
519 * We allocate a buffer, fake it as if it was a pointer to a resource in a module,
520 * pass it to LookupIconIdFromDirectoryEx and get back the index we have to use
521 */
522 fakeDir = HeapAlloc(GetProcessHeap(), 0, FIELD_OFFSET(CURSORICONDIR, idEntries[dir->idCount]));
523 if(!fakeDir)
524 {
525 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
526 return NULL;
527 }
528 fakeDir->idReserved = 0;
529 fakeDir->idType = dir->idType;
530 fakeDir->idCount = dir->idCount;
531 for(i = 0; i<dir->idCount; i++)
532 {
533 fakeEntry = &fakeDir->idEntries[i];
534 entry = &dir->idEntries[i];
535 /* Take this as an occasion to perform a size check */
536 if ((entry->dwDIBOffset > dwFileSize)
537 || ((entry->dwDIBOffset + entry->dwDIBSize) > dwFileSize))
538 {
539 ERR("Corrupted icon file?.\n");
540 HeapFree(GetProcessHeap(), 0, fakeDir);
541 return NULL;
542 }
543 /* File icon/cursors are not like resource ones */
544 if(bIcon)
545 {
546 fakeEntry->ResInfo.icon.bWidth = entry->bWidth;
547 fakeEntry->ResInfo.icon.bHeight = entry->bHeight;
548 fakeEntry->ResInfo.icon.bColorCount = 0;
549 fakeEntry->ResInfo.icon.bReserved = 0;
550 }
551 else
552 {
553 fakeEntry->ResInfo.cursor.wWidth = entry->bWidth;
554 fakeEntry->ResInfo.cursor.wHeight = entry->bHeight;
555 }
556 /* Let's assume there's always one plane */
557 fakeEntry->wPlanes = 1;
558 /* We must get the bitcount from the BITMAPINFOHEADER itself */
559 if (((BITMAPINFOHEADER *)((char *)dir + entry->dwDIBOffset))->biSize == sizeof(BITMAPCOREHEADER))
560 fakeEntry->wBitCount = ((BITMAPCOREHEADER *)((char *)dir + entry->dwDIBOffset))->bcBitCount;
561 else
562 fakeEntry->wBitCount = ((BITMAPINFOHEADER *)((char *)dir + entry->dwDIBOffset))->biBitCount;
563 fakeEntry->dwBytesInRes = entry->dwDIBSize;
564 fakeEntry->wResId = i + 1;
565 }
566
567 /* Now call LookupIconIdFromResourceEx */
568 i = LookupIconIdFromDirectoryEx((PBYTE)fakeDir, bIcon, cxDesired, cyDesired, fuLoad & LR_MONOCHROME);
569 /* We don't need this anymore */
570 HeapFree(GetProcessHeap(), 0, fakeDir);
571 if(i == 0)
572 {
573 WARN("Unable to get a fit entry index.\n");
574 return NULL;
575 }
576
577 /* We found it */
578 return &dir->idEntries[i-1];
579 }
580
581 DWORD
get_best_icon_file_offset(_In_ const LPBYTE dir,_In_ DWORD dwFileSize,_In_ int cxDesired,_In_ int cyDesired,_In_ BOOL bIcon,_In_ DWORD fuLoad,_Out_ POINT * ptHotSpot)582 get_best_icon_file_offset(
583 _In_ const LPBYTE dir,
584 _In_ DWORD dwFileSize,
585 _In_ int cxDesired,
586 _In_ int cyDesired,
587 _In_ BOOL bIcon,
588 _In_ DWORD fuLoad,
589 _Out_ POINT *ptHotSpot
590 )
591 {
592 const CURSORICONFILEDIRENTRY *entry;
593
594 entry = get_best_icon_file_entry((CURSORICONFILEDIR *) dir, dwFileSize, cxDesired, cyDesired, bIcon, fuLoad);
595
596 if(ptHotSpot)
597 {
598 ptHotSpot->x = entry->xHotspot;
599 ptHotSpot->y = entry->yHotspot;
600 }
601
602 if(entry)
603 return entry->dwDIBOffset;
604
605 return 0;
606 }
607
608
609
610 /************* IMPLEMENTATION CORE ****************/
611
CURSORICON_GetCursorDataFromBMI(_Inout_ CURSORDATA * pdata,_In_ const BITMAPINFO * pbmi)612 static BOOL CURSORICON_GetCursorDataFromBMI(
613 _Inout_ CURSORDATA* pdata,
614 _In_ const BITMAPINFO *pbmi
615 )
616 {
617 UINT ubmiSize = bitmap_info_size(pbmi, DIB_RGB_COLORS);
618 BOOL monochrome = is_dib_monochrome(pbmi);
619 LONG width, height;
620 WORD bpp;
621 DWORD compr;
622 int ibmpType;
623 HDC hdc, hdcScreen;
624 BITMAPINFO* pbmiCopy;
625 HBITMAP hbmpOld = NULL;
626 BOOL bResult = FALSE;
627 const VOID *pvColor, *pvMask;
628
629 ibmpType = DIB_GetBitmapInfo(&pbmi->bmiHeader, &width, &height, &bpp, &compr);
630 /* Invalid data */
631 if(ibmpType < 0)
632 return FALSE;
633
634 /* No compression for icons */
635 if(compr != BI_RGB)
636 return FALSE;
637
638 /* If no dimensions were set, use the one from the icon */
639 if(!pdata->cx) pdata->cx = width;
640 if(!pdata->cy) pdata->cy = height < 0 ? -height/2 : height/2;
641
642 /* Fix the hotspot coords */
643 if(pdata->rt == (USHORT)((ULONG_PTR)RT_CURSOR))
644 {
645 if(pdata->cx != width)
646 pdata->xHotspot = (pdata->xHotspot * pdata->cx) / width;
647 if(pdata->cy != height/2)
648 pdata->yHotspot = (pdata->yHotspot * pdata->cy * 2) / height;
649 }
650 else
651 {
652 pdata->xHotspot = pdata->cx/2;
653 pdata->yHotspot = pdata->cy/2;
654 }
655
656 hdcScreen = CreateDCW(DISPLAYW, NULL, NULL, NULL);
657 if(!hdcScreen)
658 return FALSE;
659 hdc = CreateCompatibleDC(hdcScreen);
660 if(!hdc)
661 {
662 DeleteDC(hdcScreen);
663 return FALSE;
664 }
665
666 pbmiCopy = HeapAlloc(GetProcessHeap(), 0, max(ubmiSize, FIELD_OFFSET(BITMAPINFO, bmiColors[3])));
667 if(!pbmiCopy)
668 goto done;
669 RtlCopyMemory(pbmiCopy, pbmi, ubmiSize);
670
671 /* In an icon/cursor, the BITMAPINFO holds twice the height */
672 if(pbmiCopy->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
673 ((BITMAPCOREHEADER*)&pbmiCopy->bmiHeader)->bcHeight /= 2;
674 else
675 pbmiCopy->bmiHeader.biHeight /= 2;
676 height /= 2;
677
678 pvColor = (const char*)pbmi + ubmiSize;
679 pvMask = (const char*)pvColor +
680 get_dib_image_size(width, height, bpp );
681
682 /* Set XOR bits */
683 if(monochrome)
684 {
685 /* Create the 1bpp bitmap which will contain everything */
686 pdata->hbmColor = NULL;
687 pdata->hbmMask = CreateBitmap(pdata->cx, pdata->cy * 2, 1, 1, NULL);
688 if(!pdata->hbmMask)
689 goto done;
690 hbmpOld = SelectObject(hdc, pdata->hbmMask);
691 if(!hbmpOld)
692 goto done;
693
694 if(!StretchDIBits(hdc, 0, pdata->cy, pdata->cx, pdata->cy,
695 0, 0, width, height,
696 pvColor, pbmiCopy, DIB_RGB_COLORS, SRCCOPY))
697 goto done;
698 pdata->bpp = 1;
699 }
700 else
701 {
702 /* Create the bitmap. It has to be compatible with the screen surface */
703 pdata->hbmColor = CreateCompatibleBitmap(hdcScreen, pdata->cx, pdata->cy);
704 if(!pdata->hbmColor)
705 goto done;
706 /* Create the 1bpp mask bitmap */
707 pdata->hbmMask = CreateBitmap(pdata->cx, pdata->cy, 1, 1, NULL);
708 if(!pdata->hbmMask)
709 goto done;
710 hbmpOld = SelectObject(hdc, pdata->hbmColor);
711 if(!hbmpOld)
712 goto done;
713 if(!StretchDIBits(hdc, 0, 0, pdata->cx, pdata->cy,
714 0, 0, width, height,
715 pvColor, pbmiCopy, DIB_RGB_COLORS, SRCCOPY))
716 goto done;
717 pdata->bpp = GetDeviceCaps(hdcScreen, BITSPIXEL);
718 pdata->hbmAlpha = create_alpha_bitmap(NULL, pbmiCopy, pvColor, pdata->cx, pdata->cy);
719
720 /* Now convert the info to monochrome for the mask bits */
721 if (pbmiCopy->bmiHeader.biSize != sizeof(BITMAPCOREHEADER))
722 {
723 RGBQUAD *rgb = pbmiCopy->bmiColors;
724
725 pbmiCopy->bmiHeader.biClrUsed = pbmiCopy->bmiHeader.biClrImportant = 2;
726 rgb[0].rgbBlue = rgb[0].rgbGreen = rgb[0].rgbRed = 0x00;
727 rgb[1].rgbBlue = rgb[1].rgbGreen = rgb[1].rgbRed = 0xff;
728 rgb[0].rgbReserved = rgb[1].rgbReserved = 0;
729 pbmiCopy->bmiHeader.biBitCount = 1;
730 }
731 else
732 {
733 RGBTRIPLE *rgb = (RGBTRIPLE *)(((BITMAPCOREHEADER *)pbmiCopy) + 1);
734
735 rgb[0].rgbtBlue = rgb[0].rgbtGreen = rgb[0].rgbtRed = 0x00;
736 rgb[1].rgbtBlue = rgb[1].rgbtGreen = rgb[1].rgbtRed = 0xff;
737 ((BITMAPCOREHEADER*)&pbmiCopy->bmiHeader)->bcBitCount = 1;
738 }
739 }
740 /* Set the mask bits */
741 if(!SelectObject(hdc, pdata->hbmMask))
742 goto done;
743 bResult = StretchDIBits(hdc, 0, 0, pdata->cx, pdata->cy,
744 0, 0, width, height,
745 pvMask, pbmiCopy, DIB_RGB_COLORS, SRCCOPY) != 0;
746
747 done:
748 DeleteDC(hdcScreen);
749 if(hbmpOld) SelectObject(hdc, hbmpOld);
750 DeleteDC(hdc);
751 if(pbmiCopy) HeapFree(GetProcessHeap(), 0, pbmiCopy);
752 /* Clean up in case of failure */
753 if(!bResult)
754 {
755 if(pdata->hbmMask) DeleteObject(pdata->hbmMask);
756 if(pdata->hbmColor) DeleteObject(pdata->hbmColor);
757 if(pdata->hbmAlpha) DeleteObject(pdata->hbmAlpha);
758 }
759 return bResult;
760 }
761
CURSORICON_GetCursorDataFromIconInfo(_Out_ CURSORDATA * pCursorData,_In_ ICONINFO * pIconInfo)762 static BOOL CURSORICON_GetCursorDataFromIconInfo(
763 _Out_ CURSORDATA* pCursorData,
764 _In_ ICONINFO* pIconInfo
765 )
766 {
767 BITMAP bm;
768
769 ZeroMemory(pCursorData, sizeof(*pCursorData));
770 if(pIconInfo->hbmColor)
771 {
772 /* We must convert the color bitmap to screen format */
773 HDC hdcScreen, hdcMem;
774 HBITMAP hbmpPrev;
775
776 /* The mask dictates its dimensions */
777 if (!GetObject(pIconInfo->hbmMask, sizeof(bm), &bm))
778 return FALSE;
779 hdcScreen = CreateDCW(DISPLAYW, NULL, NULL, NULL);
780 if(!hdcScreen)
781 return FALSE;
782 hdcMem = CreateCompatibleDC(hdcScreen);
783 if(!hdcMem)
784 {
785 DeleteDC(hdcScreen);
786 return FALSE;
787 }
788 pCursorData->hbmColor = CreateCompatibleBitmap(hdcScreen, bm.bmWidth, bm.bmHeight);
789 DeleteDC(hdcScreen);
790 if (!pCursorData->hbmColor)
791 {
792 DeleteDC(hdcMem);
793 return FALSE;
794 }
795 hbmpPrev = SelectObject(hdcMem, pCursorData->hbmColor);
796 if (!hbmpPrev)
797 {
798 DeleteDC(hdcMem);
799 DeleteObject(pCursorData->hbmColor);
800 return FALSE;
801 }
802 stretch_blt_icon( hdcMem, bm.bmWidth, bm.bmHeight, pIconInfo->hbmColor);
803 SelectObject(hdcMem, hbmpPrev);
804 DeleteDC(hdcMem);
805 }
806 pCursorData->hbmMask = CopyImage(pIconInfo->hbmMask, IMAGE_BITMAP, 0, 0, LR_MONOCHROME);
807 if(!pCursorData->hbmMask)
808 return FALSE;
809
810 /* Now, fill some information */
811 pCursorData->rt = (USHORT)((ULONG_PTR)(pIconInfo->fIcon ? RT_ICON : RT_CURSOR));
812 if(pCursorData->hbmColor)
813 {
814 GetObject(pCursorData->hbmColor, sizeof(bm), &bm);
815 pCursorData->bpp = bm.bmBitsPixel;
816 pCursorData->cx = bm.bmWidth;
817 pCursorData->cy = bm.bmHeight;
818 if(pCursorData->bpp == 32)
819 pCursorData->hbmAlpha = create_alpha_bitmap(pCursorData->hbmColor, NULL, NULL, 0, 0);
820 }
821 else
822 {
823 GetObject(pCursorData->hbmMask, sizeof(bm), &bm);
824 pCursorData->bpp = 1;
825 pCursorData->cx = bm.bmWidth;
826 pCursorData->cy = bm.bmHeight/2;
827 }
828
829 if(pIconInfo->fIcon)
830 {
831 pCursorData->xHotspot = pCursorData->cx/2;
832 pCursorData->yHotspot = pCursorData->cy/2;
833 }
834 else
835 {
836 pCursorData->xHotspot = pIconInfo->xHotspot;
837 pCursorData->yHotspot = pIconInfo->yHotspot;
838 }
839
840 return TRUE;
841 }
842
843
844 #define RIFF_FOURCC( c0, c1, c2, c3 ) \
845 ( (DWORD)(BYTE)(c0) | ( (DWORD)(BYTE)(c1) << 8 ) | \
846 ( (DWORD)(BYTE)(c2) << 16 ) | ( (DWORD)(BYTE)(c3) << 24 ) )
847
848 #define ANI_RIFF_ID RIFF_FOURCC('R', 'I', 'F', 'F')
849 #define ANI_LIST_ID RIFF_FOURCC('L', 'I', 'S', 'T')
850 #define ANI_ACON_ID RIFF_FOURCC('A', 'C', 'O', 'N')
851 #define ANI_anih_ID RIFF_FOURCC('a', 'n', 'i', 'h')
852 #define ANI_seq__ID RIFF_FOURCC('s', 'e', 'q', ' ')
853 #define ANI_fram_ID RIFF_FOURCC('f', 'r', 'a', 'm')
854 #define ANI_rate_ID RIFF_FOURCC('r', 'a', 't', 'e')
855
856 #define ANI_FLAG_ICON 0x1
857 #define ANI_FLAG_SEQUENCE 0x2
858
859 #include <pshpack1.h>
860 typedef struct {
861 DWORD header_size;
862 DWORD num_frames;
863 DWORD num_steps;
864 DWORD width;
865 DWORD height;
866 DWORD bpp;
867 DWORD num_planes;
868 DWORD display_rate;
869 DWORD flags;
870 } ani_header;
871
872 typedef struct {
873 DWORD data_size;
874 const unsigned char *data;
875 } riff_chunk_t;
876 #include <poppack.h>
877
dump_ani_header(const ani_header * header)878 static void dump_ani_header( const ani_header *header )
879 {
880 TRACE(" header size: %d\n", header->header_size);
881 TRACE(" frames: %d\n", header->num_frames);
882 TRACE(" steps: %d\n", header->num_steps);
883 TRACE(" width: %d\n", header->width);
884 TRACE(" height: %d\n", header->height);
885 TRACE(" bpp: %d\n", header->bpp);
886 TRACE(" planes: %d\n", header->num_planes);
887 TRACE(" display rate: %d\n", header->display_rate);
888 TRACE(" flags: 0x%08x\n", header->flags);
889 }
890
891 /* Find an animated cursor chunk, given its type and ID */
riff_find_chunk(DWORD chunk_id,DWORD chunk_type,const riff_chunk_t * parent_chunk,riff_chunk_t * chunk)892 static void riff_find_chunk( DWORD chunk_id, DWORD chunk_type, const riff_chunk_t *parent_chunk, riff_chunk_t *chunk )
893 {
894 const unsigned char *ptr = parent_chunk->data;
895 const unsigned char *end = parent_chunk->data + (parent_chunk->data_size - (2 * sizeof(DWORD)));
896
897 if (chunk_type == ANI_LIST_ID || chunk_type == ANI_RIFF_ID) end -= sizeof(DWORD);
898
899 while (ptr < end)
900 {
901 if ((!chunk_type && *(const DWORD *)ptr == chunk_id )
902 || (chunk_type && *(const DWORD *)ptr == chunk_type && *((const DWORD *)ptr + 2) == chunk_id ))
903 {
904 ptr += sizeof(DWORD);
905 chunk->data_size = (*(const DWORD *)ptr + 1) & ~1;
906 ptr += sizeof(DWORD);
907 if (chunk_type == ANI_LIST_ID || chunk_type == ANI_RIFF_ID) ptr += sizeof(DWORD);
908 chunk->data = ptr;
909
910 return;
911 }
912
913 ptr += sizeof(DWORD);
914 ptr += (*(const DWORD *)ptr + 1) & ~1;
915 ptr += sizeof(DWORD);
916 }
917 }
918
CURSORICON_GetCursorDataFromANI(_Inout_ CURSORDATA * pCurData,_In_ const BYTE * pData,_In_ DWORD dwDataSize,_In_ DWORD fuLoad)919 static BOOL CURSORICON_GetCursorDataFromANI(
920 _Inout_ CURSORDATA* pCurData,
921 _In_ const BYTE *pData,
922 _In_ DWORD dwDataSize,
923 _In_ DWORD fuLoad
924 )
925 {
926 UINT i;
927 const ani_header *pHeader;
928 riff_chunk_t root_chunk = { dwDataSize, pData };
929 riff_chunk_t ACON_chunk = {0};
930 riff_chunk_t anih_chunk = {0};
931 riff_chunk_t fram_chunk = {0};
932 riff_chunk_t rate_chunk = {0};
933 riff_chunk_t seq_chunk = {0};
934 const unsigned char *icon_chunk;
935 const unsigned char *icon_data;
936
937 /* Find the root chunk */
938 riff_find_chunk( ANI_ACON_ID, ANI_RIFF_ID, &root_chunk, &ACON_chunk );
939 if (!ACON_chunk.data)
940 {
941 ERR("Failed to get root chunk.\n");
942 return FALSE;
943 }
944
945 /* Find the header chunk */
946 riff_find_chunk( ANI_anih_ID, 0, &ACON_chunk, &anih_chunk );
947 if (!ACON_chunk.data)
948 {
949 ERR("Failed to get header chunk.\n");
950 return FALSE;
951 }
952 pHeader = (ani_header*)anih_chunk.data;
953 dump_ani_header(pHeader);
954
955 /* Set up the master data */
956 pCurData->CURSORF_flags |= CURSORF_ACON;
957 pCurData->cpcur = pHeader->num_frames;
958 pCurData->cicur = pHeader->num_steps;
959 pCurData->iicur = pHeader->display_rate;
960
961 /* Get the sequences */
962 if (pHeader->flags & ANI_FLAG_SEQUENCE)
963 {
964 riff_find_chunk( ANI_seq__ID, 0, &ACON_chunk, &seq_chunk );
965 if (!seq_chunk.data)
966 {
967 ERR("No sequence data although the flag is set!\n");
968 return FALSE;
969 }
970 }
971
972 /* Get the frame rates */
973 riff_find_chunk( ANI_rate_ID, 0, &ACON_chunk, &rate_chunk );
974 if (rate_chunk.data)
975 pCurData->ajifRate = (INT*)rate_chunk.data;
976
977 /* Get the frames chunk */
978 riff_find_chunk( ANI_fram_ID, ANI_LIST_ID, &ACON_chunk, &fram_chunk );
979 if (!fram_chunk.data)
980 {
981 ERR("Failed to get icon list.\n");
982 return 0;
983 }
984 icon_chunk = fram_chunk.data;
985 icon_data = fram_chunk.data + (2 * sizeof(DWORD));
986
987 if(pHeader->num_frames > 1)
988 {
989 /* Allocate frame descriptors, step indices and rates */
990 pCurData->aspcur = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
991 pHeader->num_frames * sizeof(CURSORDATA) + pHeader->num_steps * (sizeof(DWORD) + sizeof(INT)));
992 if(!pCurData->aspcur)
993 {
994 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
995 return FALSE;
996 }
997 pCurData->aicur = (DWORD*)(pCurData->aspcur + pHeader->num_frames);
998 pCurData->ajifRate = (INT*)(pCurData->aicur + pHeader->num_steps);
999 }
1000
1001 for(i=0; i < pHeader->num_frames; i++)
1002 {
1003 CURSORDATA* pFrameData;
1004 const DWORD chunk_size = *(const DWORD *)(icon_chunk + sizeof(DWORD));
1005 const BITMAPINFO* pbmi;
1006
1007 if(pHeader->num_frames > 1)
1008 pFrameData = &pCurData->aspcur[i];
1009 else
1010 pFrameData = pCurData;
1011
1012 pFrameData->rt = pCurData->rt;
1013
1014 if (pHeader->flags & ANI_FLAG_ICON)
1015 {
1016 /* The chunks describe an icon file */
1017 const CURSORICONFILEDIRENTRY* pDirEntry = get_best_icon_file_entry(
1018 (const CURSORICONFILEDIR *) icon_data,
1019 chunk_size,
1020 pCurData->cx,
1021 pCurData->cy,
1022 TRUE,
1023 fuLoad);
1024 if(!pDirEntry)
1025 {
1026 ERR("Unable to find the right file entry for frame %d.\n", i);
1027 goto error;
1028 }
1029 pFrameData->xHotspot = pDirEntry->xHotspot;
1030 pFrameData->yHotspot = pDirEntry->yHotspot;
1031 if(!pHeader->width || !pHeader->height)
1032 {
1033 pFrameData->cx = pDirEntry->bWidth;
1034 pFrameData->cy = pDirEntry->bHeight;
1035 }
1036 else
1037 {
1038 pFrameData->cx = pHeader->width;
1039 pFrameData->cy = pHeader->height;
1040 }
1041 pbmi = (const BITMAPINFO *) (icon_data + pDirEntry->dwDIBOffset);
1042 }
1043 else
1044 {
1045 /* The chunks just describe bitmaps */
1046 pbmi = (const BITMAPINFO *)icon_data;
1047 pFrameData->xHotspot = pFrameData->yHotspot = 0;
1048 }
1049
1050 /* Do the real work */
1051 CURSORICON_GetCursorDataFromBMI(pFrameData, pbmi);
1052
1053 if(pHeader->num_frames > 1)
1054 pFrameData->CURSORF_flags |= CURSORF_ACONFRAME;
1055 else
1056 pFrameData->CURSORF_flags &= ~CURSORF_ACON;
1057
1058
1059 /* Next frame */
1060 icon_chunk += chunk_size + (2 * sizeof(DWORD));
1061 icon_data = icon_chunk + (2 * sizeof(DWORD));
1062 }
1063
1064 if(pHeader->num_frames <= 1)
1065 return TRUE;
1066
1067 if(rate_chunk.data)
1068 CopyMemory(pCurData->ajifRate, rate_chunk.data, pHeader->num_steps * sizeof(INT));
1069 else
1070 {
1071 for(i=0; i < pHeader->num_steps; i++)
1072 pCurData->ajifRate[i] = pHeader->display_rate;
1073 }
1074
1075 if (pHeader->flags & ANI_FLAG_SEQUENCE)
1076 {
1077 CopyMemory(pCurData->aicur, seq_chunk.data, pHeader->num_steps * sizeof(DWORD));
1078 }
1079 else
1080 {
1081 for(i=0; i < pHeader->num_steps; i++)
1082 pCurData->aicur[i] = i;
1083 }
1084
1085 return TRUE;
1086
1087 error:
1088 HeapFree(GetProcessHeap(), 0, pCurData->aspcur);
1089 ZeroMemory(pCurData, sizeof(CURSORDATA));
1090 return FALSE;
1091 }
1092
1093
1094 static
1095 HBITMAP
BITMAP_LoadImageW(_In_opt_ HINSTANCE hinst,_In_ LPCWSTR lpszName,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuLoad)1096 BITMAP_LoadImageW(
1097 _In_opt_ HINSTANCE hinst,
1098 _In_ LPCWSTR lpszName,
1099 _In_ int cxDesired,
1100 _In_ int cyDesired,
1101 _In_ UINT fuLoad
1102 )
1103 {
1104 const BITMAPINFO* pbmi;
1105 BITMAPINFO* pbmiScaled = NULL;
1106 BITMAPINFO* pbmiCopy = NULL;
1107 const VOID* pvMapping = NULL;
1108 DWORD dwOffset = 0;
1109 HGLOBAL hgRsrc = NULL;
1110 int iBMISize;
1111 PVOID pvBits;
1112 HDC hdcScreen = NULL;
1113 HDC hdc = NULL;
1114 HBITMAP hbmpOld, hbmpRet = NULL;
1115 LONG width, height;
1116 WORD bpp;
1117 DWORD compr, ResSize = 0;
1118
1119 /* Map the bitmap info */
1120 if(fuLoad & LR_LOADFROMFILE)
1121 {
1122 const BITMAPFILEHEADER* pbmfh;
1123
1124 pvMapping = map_fileW(lpszName, NULL);
1125 if(!pvMapping)
1126 return NULL;
1127 pbmfh = pvMapping;
1128 if (pbmfh->bfType != 0x4d42 /* 'BM' */)
1129 {
1130 WARN("Invalid/unsupported bitmap format!\n");
1131 goto end;
1132 }
1133 pbmi = (const BITMAPINFO*)(pbmfh + 1);
1134
1135 /* Get the image bits */
1136 if(pbmfh->bfOffBits)
1137 dwOffset = pbmfh->bfOffBits - sizeof(BITMAPFILEHEADER);
1138 }
1139 else
1140 {
1141 HRSRC hrsrc;
1142
1143 /* Caller wants an OEM bitmap */
1144 if(!hinst)
1145 hinst = User32Instance;
1146 hrsrc = FindResourceW(hinst, lpszName, RT_BITMAP);
1147 if(!hrsrc)
1148 return NULL;
1149 hgRsrc = LoadResource(hinst, hrsrc);
1150 if(!hgRsrc)
1151 return NULL;
1152 pbmi = LockResource(hgRsrc);
1153 if(!pbmi)
1154 return NULL;
1155 ResSize = SizeofResource(hinst, hrsrc);
1156 }
1157
1158 /* Fix up values */
1159 if(DIB_GetBitmapInfo(&pbmi->bmiHeader, &width, &height, &bpp, &compr) == -1)
1160 goto end;
1161 if((width > 65535) || (height > 65535))
1162 goto end;
1163 if(cxDesired == 0)
1164 cxDesired = width;
1165 if(cyDesired == 0)
1166 cyDesired = height;
1167 else if(height < 0)
1168 cyDesired = -cyDesired;
1169
1170 iBMISize = bitmap_info_size(pbmi, DIB_RGB_COLORS);
1171
1172 /* Get a pointer to the image data */
1173 pvBits = (char*)pbmi + (dwOffset ? dwOffset : iBMISize);
1174
1175 /* Create a copy of the info describing the bitmap in the file */
1176 pbmiCopy = HeapAlloc(GetProcessHeap(), 0, iBMISize);
1177 if(!pbmiCopy)
1178 goto end;
1179 CopyMemory(pbmiCopy, pbmi, iBMISize);
1180
1181 TRACE("Size Image %d, Size Header %d, ResSize %d\n",
1182 pbmiCopy->bmiHeader.biSizeImage, pbmiCopy->bmiHeader.biSize, ResSize);
1183
1184 /* HACK: If this is a binutils' windres.exe compiled 16 or 32 bpp bitmap
1185 * using BI_BITFIELDS, then a bug causes it to fail to include
1186 * the bytes for the bitfields. So, we have to substract out the
1187 * size of the bitfields previously included from bitmap_info_size. */
1188 if (compr == BI_BITFIELDS && (bpp == 16 || bpp == 32) &&
1189 pbmiCopy->bmiHeader.biSizeImage + pbmiCopy->bmiHeader.biSize == ResSize)
1190 {
1191 /* GCC pointer to the image data has 12 less bytes than MSVC */
1192 pvBits = (char*)pvBits - 12;
1193 WARN("Found GCC Resource Compiled 16-bpp or 32-bpp error\n");
1194 }
1195
1196 /* Fix it up, if needed */
1197 if(fuLoad & (LR_LOADTRANSPARENT | LR_LOADMAP3DCOLORS))
1198 {
1199 WORD bpp, incr, numColors;
1200 char* pbmiColors;
1201 RGBTRIPLE* ptr;
1202 COLORREF crWindow, cr3DShadow, cr3DFace, cr3DLight;
1203 BYTE pixel = *((BYTE*)pvBits);
1204 UINT i;
1205
1206 if(pbmiCopy->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
1207 {
1208 bpp = ((BITMAPCOREHEADER*)&pbmiCopy->bmiHeader)->bcBitCount;
1209 numColors = 1 << bpp;
1210 /* BITMAPCOREINFO holds RGBTRIPLEs */
1211 incr = 3;
1212 }
1213 else
1214 {
1215 bpp = pbmiCopy->bmiHeader.biBitCount;
1216 /* BITMAPINFOHEADER holds RGBQUADs */
1217 incr = 4;
1218 numColors = pbmiCopy->bmiHeader.biClrUsed;
1219 if(numColors > 256) numColors = 256;
1220 if (!numColors && (bpp <= 8)) numColors = 1 << bpp;
1221 }
1222
1223 if(bpp > 8)
1224 goto create_bitmap;
1225
1226 pbmiColors = (char*)pbmiCopy + pbmiCopy->bmiHeader.biSize;
1227
1228 /* Get the relevant colors */
1229 crWindow = GetSysColor(COLOR_WINDOW);
1230 cr3DShadow = GetSysColor(COLOR_3DSHADOW);
1231 cr3DFace = GetSysColor(COLOR_3DFACE);
1232 cr3DLight = GetSysColor(COLOR_3DLIGHT);
1233
1234 /* Fix the transparent palette entry */
1235 if(fuLoad & LR_LOADTRANSPARENT)
1236 {
1237 switch(bpp)
1238 {
1239 case 1: pixel >>= 7; break;
1240 case 4: pixel >>= 4; break;
1241 case 8: break;
1242 default:
1243 FIXME("Unhandled bit depth %d.\n", bpp);
1244 goto create_bitmap;
1245 }
1246
1247 if(pixel >= numColors)
1248 {
1249 ERR("Wrong pixel passed in.\n");
1250 goto create_bitmap;
1251 }
1252
1253 /* If both flags are set, we must use COLOR_3DFACE */
1254 if(fuLoad & LR_LOADMAP3DCOLORS) crWindow = cr3DFace;
1255
1256 /* Define the color */
1257 ptr = (RGBTRIPLE*)(pbmiColors + pixel*incr);
1258 ptr->rgbtBlue = GetBValue(crWindow);
1259 ptr->rgbtGreen = GetGValue(crWindow);
1260 ptr->rgbtRed = GetRValue(crWindow);
1261 goto create_bitmap;
1262 }
1263
1264 /* If we are here, then LR_LOADMAP3DCOLORS is set without LR_TRANSPARENT */
1265 for(i = 0; i<numColors; i++)
1266 {
1267 ptr = (RGBTRIPLE*)(pbmiColors + i*incr);
1268 if((ptr->rgbtBlue == ptr->rgbtRed) && (ptr->rgbtBlue == ptr->rgbtGreen))
1269 {
1270 if(ptr->rgbtBlue == 128)
1271 {
1272 ptr->rgbtBlue = GetBValue(cr3DShadow);
1273 ptr->rgbtGreen = GetGValue(cr3DShadow);
1274 ptr->rgbtRed = GetRValue(cr3DShadow);
1275 }
1276 if(ptr->rgbtBlue == 192)
1277 {
1278 ptr->rgbtBlue = GetBValue(cr3DFace);
1279 ptr->rgbtGreen = GetGValue(cr3DFace);
1280 ptr->rgbtRed = GetRValue(cr3DFace);
1281 }
1282 if(ptr->rgbtBlue == 223)
1283 {
1284 ptr->rgbtBlue = GetBValue(cr3DLight);
1285 ptr->rgbtGreen = GetGValue(cr3DLight);
1286 ptr->rgbtRed = GetRValue(cr3DLight);
1287 }
1288 }
1289 }
1290 }
1291
1292 create_bitmap:
1293 if(fuLoad & LR_CREATEDIBSECTION)
1294 {
1295 /* Allocate the BMI describing the new bitmap */
1296 pbmiScaled = HeapAlloc(GetProcessHeap(), 0, iBMISize);
1297 if(!pbmiScaled)
1298 goto end;
1299 CopyMemory(pbmiScaled, pbmiCopy, iBMISize);
1300
1301 /* Fix it up */
1302 if(pbmiScaled->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
1303 {
1304 BITMAPCOREHEADER* pbmch = (BITMAPCOREHEADER*)&pbmiScaled->bmiHeader;
1305 pbmch->bcWidth = cxDesired;
1306 pbmch->bcHeight = cyDesired;
1307 }
1308 else
1309 {
1310 pbmiScaled->bmiHeader.biWidth = cxDesired;
1311 pbmiScaled->bmiHeader.biHeight = cyDesired;
1312 /* No compression for DIB sections */
1313 pbmiScaled->bmiHeader.biCompression = BI_RGB;
1314 }
1315 }
1316
1317 /* Top-down image */
1318 if(cyDesired < 0) cyDesired = -cyDesired;
1319
1320 /* We need a device context */
1321 hdcScreen = CreateDCW(DISPLAYW, NULL, NULL, NULL);
1322 if(!hdcScreen)
1323 goto end;
1324 hdc = CreateCompatibleDC(hdcScreen);
1325 if(!hdc)
1326 goto end;
1327
1328 /* Now create the bitmap */
1329 if(fuLoad & LR_CREATEDIBSECTION)
1330 hbmpRet = CreateDIBSection(hdc, pbmiScaled, DIB_RGB_COLORS, NULL, 0, 0);
1331 else
1332 {
1333 if(is_dib_monochrome(pbmiCopy) || (fuLoad & LR_MONOCHROME))
1334 hbmpRet = CreateBitmap(cxDesired, cyDesired, 1, 1, NULL);
1335 else
1336 hbmpRet = CreateCompatibleBitmap(hdcScreen, cxDesired, cyDesired);
1337 }
1338
1339 if(!hbmpRet)
1340 goto end;
1341
1342 hbmpOld = SelectObject(hdc, hbmpRet);
1343 if(!hbmpOld)
1344 goto end;
1345 if(!StretchDIBits(hdc, 0, 0, cxDesired, cyDesired,
1346 0, 0, width, height,
1347 pvBits, pbmiCopy, DIB_RGB_COLORS, SRCCOPY))
1348 {
1349 ERR("StretchDIBits failed!.\n");
1350 SelectObject(hdc, hbmpOld);
1351 DeleteObject(hbmpRet);
1352 hbmpRet = NULL;
1353 goto end;
1354 }
1355
1356 SelectObject(hdc, hbmpOld);
1357
1358 end:
1359 if(hdcScreen)
1360 DeleteDC(hdcScreen);
1361 if(hdc)
1362 DeleteDC(hdc);
1363 if(pbmiScaled)
1364 HeapFree(GetProcessHeap(), 0, pbmiScaled);
1365 if(pbmiCopy)
1366 HeapFree(GetProcessHeap(), 0, pbmiCopy);
1367 if (pvMapping)
1368 UnmapViewOfFile( pvMapping );
1369 if(hgRsrc)
1370 FreeResource(hgRsrc);
1371
1372 return hbmpRet;
1373 }
1374
1375
1376 static
1377 HANDLE
CURSORICON_LoadFromFileW(_In_ LPCWSTR lpszName,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuLoad,_In_ BOOL bIcon)1378 CURSORICON_LoadFromFileW(
1379 _In_ LPCWSTR lpszName,
1380 _In_ int cxDesired,
1381 _In_ int cyDesired,
1382 _In_ UINT fuLoad,
1383 _In_ BOOL bIcon
1384 )
1385 {
1386 const CURSORICONFILEDIRENTRY *entry;
1387 const CURSORICONFILEDIR *dir;
1388 DWORD filesize = 0;
1389 LPBYTE bits;
1390 HANDLE hCurIcon = NULL;
1391 CURSORDATA cursorData;
1392
1393 TRACE("loading %s\n", debugstr_w( lpszName ));
1394
1395 bits = map_fileW( lpszName, &filesize );
1396 if (!bits)
1397 return NULL;
1398
1399 /* Check for .ani. */
1400 if (memcmp( bits, "RIFF", 4 ) == 0)
1401 {
1402 UNIMPLEMENTED;
1403 goto end;
1404 }
1405
1406 dir = (CURSORICONFILEDIR*) bits;
1407 entry = get_best_icon_file_entry(dir, filesize, cxDesired, cyDesired, bIcon, fuLoad);
1408 if(!entry)
1409 goto end;
1410
1411 /* Fix dimensions */
1412 if(!cxDesired) cxDesired = entry->bWidth;
1413 if(!cyDesired) cyDesired = entry->bHeight;
1414 /* A bit of preparation */
1415 ZeroMemory(&cursorData, sizeof(cursorData));
1416 if(!bIcon)
1417 {
1418 cursorData.xHotspot = entry->xHotspot;
1419 cursorData.yHotspot = entry->yHotspot;
1420 }
1421 cursorData.rt = (USHORT)((ULONG_PTR)(bIcon ? RT_ICON : RT_CURSOR));
1422
1423 /* Do the dance */
1424 if(!CURSORICON_GetCursorDataFromBMI(&cursorData, (BITMAPINFO*)(&bits[entry->dwDIBOffset])))
1425 {
1426 ERR("Failing File is \n '%S'.\n", lpszName);
1427 goto end;
1428 }
1429
1430 hCurIcon = NtUserxCreateEmptyCurObject(FALSE);
1431 if(!hCurIcon)
1432 goto end;
1433
1434 /* Tell win32k */
1435 if(!NtUserSetCursorIconData(hCurIcon, NULL, NULL, &cursorData))
1436 {
1437 NtUserDestroyCursor(hCurIcon, TRUE);
1438 goto end_error;
1439 }
1440
1441 end:
1442 UnmapViewOfFile(bits);
1443 return hCurIcon;
1444
1445 /* Clean up */
1446 end_error:
1447 DeleteObject(cursorData.hbmMask);
1448 if(cursorData.hbmColor) DeleteObject(cursorData.hbmColor);
1449 if(cursorData.hbmAlpha) DeleteObject(cursorData.hbmAlpha);
1450 UnmapViewOfFile(bits);
1451
1452 return NULL;
1453 }
1454
1455 static
1456 HANDLE
CURSORICON_LoadImageW(_In_opt_ HINSTANCE hinst,_In_ LPCWSTR lpszName,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuLoad,_In_ BOOL bIcon)1457 CURSORICON_LoadImageW(
1458 _In_opt_ HINSTANCE hinst,
1459 _In_ LPCWSTR lpszName,
1460 _In_ int cxDesired,
1461 _In_ int cyDesired,
1462 _In_ UINT fuLoad,
1463 _In_ BOOL bIcon
1464 )
1465 {
1466 HRSRC hrsrc;
1467 HANDLE handle, hCurIcon = NULL;
1468 CURSORICONDIR* dir;
1469 WORD wResId;
1470 LPBYTE bits;
1471 CURSORDATA cursorData;
1472 BOOL bStatus;
1473 UNICODE_STRING ustrRsrc;
1474 UNICODE_STRING ustrModule = {0, 0, NULL};
1475
1476 /* Fix width/height */
1477 if(fuLoad & LR_DEFAULTSIZE)
1478 {
1479 if(!cxDesired) cxDesired = GetSystemMetrics(bIcon ? SM_CXICON : SM_CXCURSOR);
1480 if(!cyDesired) cyDesired = GetSystemMetrics(bIcon ? SM_CYICON : SM_CYCURSOR);
1481 }
1482
1483 if(fuLoad & LR_LOADFROMFILE)
1484 {
1485 return CURSORICON_LoadFromFileW(lpszName, cxDesired, cyDesired, fuLoad, bIcon);
1486 }
1487
1488 /* Check if caller wants OEM icons */
1489 if(!hinst)
1490 hinst = User32Instance;
1491
1492 if(lpszName)
1493 {
1494 /* Prepare the resource name string */
1495 if(IS_INTRESOURCE(lpszName))
1496 {
1497 ustrRsrc.Buffer = (LPWSTR)lpszName;
1498 ustrRsrc.Length = 0;
1499 ustrRsrc.MaximumLength = 0;
1500 }
1501 else
1502 RtlInitUnicodeString(&ustrRsrc, lpszName);
1503 }
1504
1505 if(LDR_IS_RESOURCE(hinst))
1506 {
1507 /* We don't have a real module for GetModuleFileName, construct a fake name instead.
1508 * GetIconInfoEx reveals the name used by Windows. */
1509 LPCWSTR fakeNameFmt = sizeof(void*) > 4 ? L"\x01%016IX" : L"\x01%08IX";
1510 ustrModule.MaximumLength = 18 * sizeof(WCHAR);
1511 ustrModule.Buffer = HeapAlloc(GetProcessHeap(), 0, ustrModule.MaximumLength);
1512 if (!ustrModule.Buffer)
1513 {
1514 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1515 return NULL;
1516 }
1517 ustrModule.Length = wsprintfW(ustrModule.Buffer, fakeNameFmt, hinst) * sizeof(WCHAR);
1518 }
1519 else if(hinst)
1520 {
1521 DWORD size = MAX_PATH;
1522 /* Get the module name string */
1523 while (TRUE)
1524 {
1525 DWORD ret;
1526 ustrModule.Buffer = HeapAlloc(GetProcessHeap(), 0, size*sizeof(WCHAR));
1527 if (!ustrModule.Buffer)
1528 {
1529 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1530 return NULL;
1531 }
1532 ret = GetModuleFileNameW(hinst, ustrModule.Buffer, size);
1533 if(ret == 0)
1534 {
1535 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1536 return NULL;
1537 }
1538
1539 /* This API is completely broken... */
1540 if (ret == size)
1541 {
1542 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1543 size *= 2;
1544 continue;
1545 }
1546
1547 ustrModule.Buffer[ret] = UNICODE_NULL;
1548 ustrModule.Length = ret * sizeof(WCHAR);
1549 ustrModule.MaximumLength = size * sizeof(WCHAR);
1550 break;
1551 }
1552 }
1553
1554 if(fuLoad & LR_SHARED)
1555 {
1556 FINDEXISTINGCURICONPARAM param;
1557
1558 TRACE("Checking for an LR_SHARED cursor/icon.\n");
1559 /* Ask win32k */
1560 param.bIcon = bIcon;
1561 param.cx = cxDesired;
1562 param.cy = cyDesired;
1563 hCurIcon = NtUserFindExistingCursorIcon(&ustrModule, &ustrRsrc, ¶m);
1564 if(hCurIcon)
1565 {
1566 /* Woohoo, got it! */
1567 TRACE("MATCH! %p\n",hCurIcon);
1568 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1569 return hCurIcon;
1570 }
1571 }
1572
1573 /* Find resource ID */
1574 hrsrc = FindResourceW(
1575 hinst,
1576 lpszName,
1577 bIcon ? RT_GROUP_ICON : RT_GROUP_CURSOR);
1578
1579 /* We let FindResource, LoadResource, etc. call SetLastError */
1580 if(!hrsrc)
1581 goto done;
1582
1583 handle = LoadResource(hinst, hrsrc);
1584 if(!handle)
1585 goto done;
1586
1587 dir = LockResource(handle);
1588 if(!dir)
1589 goto done;
1590
1591 wResId = LookupIconIdFromDirectoryEx((PBYTE)dir, bIcon, cxDesired, cyDesired, fuLoad);
1592 FreeResource(handle);
1593
1594 /* Get the relevant resource pointer */
1595 hrsrc = FindResourceW(
1596 hinst,
1597 MAKEINTRESOURCEW(wResId),
1598 bIcon ? RT_ICON : RT_CURSOR);
1599 if(!hrsrc)
1600 goto done;
1601
1602 handle = LoadResource(hinst, hrsrc);
1603 if(!handle)
1604 goto done;
1605
1606 bits = LockResource(handle);
1607 if(!bits)
1608 {
1609 FreeResource(handle);
1610 goto done;
1611 }
1612
1613 ZeroMemory(&cursorData, sizeof(cursorData));
1614
1615 /* This is from resource */
1616 cursorData.CURSORF_flags = CURSORF_FROMRESOURCE;
1617
1618 if(dir->idType == 2)
1619 {
1620 /* idType == 2 for cursor resources */
1621 SHORT* ptr = (SHORT*)bits;
1622 cursorData.xHotspot = ptr[0];
1623 cursorData.yHotspot = ptr[1];
1624 bits += 2*sizeof(SHORT);
1625 }
1626 cursorData.cx = cxDesired;
1627 cursorData.cy = cyDesired;
1628 cursorData.rt = (USHORT)((ULONG_PTR)(bIcon ? RT_ICON : RT_CURSOR));
1629
1630 /* Get the bitmaps */
1631 bStatus = CURSORICON_GetCursorDataFromBMI(
1632 &cursorData,
1633 (BITMAPINFO*)bits);
1634
1635 FreeResource( handle );
1636
1637 if(!bStatus)
1638 goto done;
1639
1640 /* Create the handle */
1641 hCurIcon = NtUserxCreateEmptyCurObject(FALSE);
1642 if(!hCurIcon)
1643 {
1644 goto end_error;
1645 }
1646
1647 if(fuLoad & LR_SHARED)
1648 {
1649 cursorData.CURSORF_flags |= CURSORF_LRSHARED;
1650 }
1651
1652 /* Tell win32k */
1653 bStatus = NtUserSetCursorIconData(hCurIcon, hinst ? &ustrModule : NULL, lpszName ? &ustrRsrc : NULL, &cursorData);
1654
1655 if(!bStatus)
1656 {
1657 NtUserDestroyCursor(hCurIcon, TRUE);
1658 goto end_error;
1659 }
1660
1661 done:
1662 if(ustrModule.Buffer)
1663 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1664 return hCurIcon;
1665
1666 end_error:
1667 if(ustrModule.Buffer)
1668 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1669 DeleteObject(cursorData.hbmMask);
1670 if(cursorData.hbmColor) DeleteObject(cursorData.hbmColor);
1671 if(cursorData.hbmAlpha) DeleteObject(cursorData.hbmAlpha);
1672
1673 return NULL;
1674 }
1675
1676 static
1677 HBITMAP
BITMAP_CopyImage(_In_ HBITMAP hnd,_In_ int desiredx,_In_ int desiredy,_In_ UINT flags)1678 BITMAP_CopyImage(
1679 _In_ HBITMAP hnd,
1680 _In_ int desiredx,
1681 _In_ int desiredy,
1682 _In_ UINT flags
1683 )
1684 {
1685 HBITMAP res = NULL;
1686 DIBSECTION ds;
1687 int objSize;
1688 BITMAPINFO * bi;
1689
1690 objSize = GetObjectW( hnd, sizeof(ds), &ds );
1691 if (!objSize) return 0;
1692 if ((desiredx < 0) || (desiredy < 0)) return 0;
1693
1694 if (flags & LR_COPYFROMRESOURCE)
1695 {
1696 FIXME("The flag LR_COPYFROMRESOURCE is not implemented for bitmaps\n");
1697 }
1698
1699 if (flags & LR_COPYRETURNORG)
1700 {
1701 FIXME("The flag LR_COPYRETURNORG is not implemented for bitmaps\n");
1702 }
1703
1704 if (desiredx == 0) desiredx = ds.dsBm.bmWidth;
1705 if (desiredy == 0) desiredy = ds.dsBm.bmHeight;
1706
1707 /* Allocate memory for a BITMAPINFOHEADER structure and a
1708 color table. The maximum number of colors in a color table
1709 is 256 which corresponds to a bitmap with depth 8.
1710 Bitmaps with higher depths don't have color tables. */
1711 bi = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD));
1712 if (!bi) return 0;
1713
1714 bi->bmiHeader.biSize = sizeof(bi->bmiHeader);
1715 bi->bmiHeader.biPlanes = ds.dsBm.bmPlanes;
1716 bi->bmiHeader.biBitCount = ds.dsBm.bmBitsPixel;
1717 bi->bmiHeader.biCompression = BI_RGB;
1718
1719 if (flags & LR_CREATEDIBSECTION)
1720 {
1721 /* Create a DIB section. LR_MONOCHROME is ignored */
1722 void * bits;
1723 HDC dc = CreateCompatibleDC(NULL);
1724
1725 if (objSize == sizeof(DIBSECTION))
1726 {
1727 /* The source bitmap is a DIB.
1728 Get its attributes to create an exact copy */
1729 memcpy(bi, &ds.dsBmih, sizeof(BITMAPINFOHEADER));
1730 }
1731
1732 bi->bmiHeader.biWidth = desiredx;
1733 bi->bmiHeader.biHeight = desiredy;
1734
1735 /* Get the color table or the color masks */
1736 GetDIBits(dc, hnd, 0, ds.dsBm.bmHeight, NULL, bi, DIB_RGB_COLORS);
1737
1738 res = CreateDIBSection(dc, bi, DIB_RGB_COLORS, &bits, NULL, 0);
1739 DeleteDC(dc);
1740 }
1741 else
1742 {
1743 /* Create a device-dependent bitmap */
1744
1745 BOOL monochrome = (flags & LR_MONOCHROME);
1746
1747 if (objSize == sizeof(DIBSECTION))
1748 {
1749 /* The source bitmap is a DIB section.
1750 Get its attributes */
1751 HDC dc = CreateCompatibleDC(NULL);
1752 bi->bmiHeader.biWidth = ds.dsBm.bmWidth;
1753 bi->bmiHeader.biHeight = ds.dsBm.bmHeight;
1754 GetDIBits(dc, hnd, 0, ds.dsBm.bmHeight, NULL, bi, DIB_RGB_COLORS);
1755 DeleteDC(dc);
1756
1757 if (!monochrome && ds.dsBm.bmBitsPixel == 1)
1758 {
1759 /* Look if the colors of the DIB are black and white */
1760
1761 monochrome =
1762 (bi->bmiColors[0].rgbRed == 0xff
1763 && bi->bmiColors[0].rgbGreen == 0xff
1764 && bi->bmiColors[0].rgbBlue == 0xff
1765 && bi->bmiColors[0].rgbReserved == 0
1766 && bi->bmiColors[1].rgbRed == 0
1767 && bi->bmiColors[1].rgbGreen == 0
1768 && bi->bmiColors[1].rgbBlue == 0
1769 && bi->bmiColors[1].rgbReserved == 0)
1770 ||
1771 (bi->bmiColors[0].rgbRed == 0
1772 && bi->bmiColors[0].rgbGreen == 0
1773 && bi->bmiColors[0].rgbBlue == 0
1774 && bi->bmiColors[0].rgbReserved == 0
1775 && bi->bmiColors[1].rgbRed == 0xff
1776 && bi->bmiColors[1].rgbGreen == 0xff
1777 && bi->bmiColors[1].rgbBlue == 0xff
1778 && bi->bmiColors[1].rgbReserved == 0);
1779 }
1780 }
1781 else if (!monochrome)
1782 {
1783 monochrome = ds.dsBm.bmBitsPixel == 1;
1784 }
1785
1786 if (monochrome)
1787 {
1788 res = CreateBitmap(desiredx, desiredy, 1, 1, NULL);
1789 }
1790 else
1791 {
1792 HDC screenDC = GetDC(NULL);
1793 res = CreateCompatibleBitmap(screenDC, desiredx, desiredy);
1794 ReleaseDC(NULL, screenDC);
1795 }
1796 }
1797
1798 if (res)
1799 {
1800 /* Only copy the bitmap if it's a DIB section or if it's
1801 compatible to the screen */
1802 BOOL copyContents;
1803
1804 if (objSize == sizeof(DIBSECTION))
1805 {
1806 copyContents = TRUE;
1807 }
1808 else
1809 {
1810 HDC screenDC = GetDC(NULL);
1811 int screen_depth = GetDeviceCaps(screenDC, BITSPIXEL);
1812 ReleaseDC(NULL, screenDC);
1813
1814 copyContents = (ds.dsBm.bmBitsPixel == 1 || ds.dsBm.bmBitsPixel == screen_depth);
1815 }
1816
1817 if (copyContents)
1818 {
1819 /* The source bitmap may already be selected in a device context,
1820 use GetDIBits/StretchDIBits and not StretchBlt */
1821
1822 HDC dc;
1823 void * bits;
1824
1825 dc = CreateCompatibleDC(NULL);
1826
1827 bi->bmiHeader.biWidth = ds.dsBm.bmWidth;
1828 bi->bmiHeader.biHeight = ds.dsBm.bmHeight;
1829 bi->bmiHeader.biSizeImage = 0;
1830 bi->bmiHeader.biClrUsed = 0;
1831 bi->bmiHeader.biClrImportant = 0;
1832
1833 /* Fill in biSizeImage */
1834 GetDIBits(dc, hnd, 0, ds.dsBm.bmHeight, NULL, bi, DIB_RGB_COLORS);
1835 bits = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, bi->bmiHeader.biSizeImage);
1836
1837 if (bits)
1838 {
1839 HBITMAP oldBmp;
1840
1841 /* Get the image bits of the source bitmap */
1842 GetDIBits(dc, hnd, 0, ds.dsBm.bmHeight, bits, bi, DIB_RGB_COLORS);
1843
1844 /* Copy it to the destination bitmap */
1845 oldBmp = SelectObject(dc, res);
1846 StretchDIBits(dc, 0, 0, desiredx, desiredy,
1847 0, 0, ds.dsBm.bmWidth, ds.dsBm.bmHeight,
1848 bits, bi, DIB_RGB_COLORS, SRCCOPY);
1849 SelectObject(dc, oldBmp);
1850
1851 HeapFree(GetProcessHeap(), 0, bits);
1852 }
1853
1854 DeleteDC(dc);
1855 }
1856
1857 if (flags & LR_COPYDELETEORG)
1858 {
1859 DeleteObject(hnd);
1860 }
1861 }
1862 HeapFree(GetProcessHeap(), 0, bi);
1863 return res;
1864 }
1865
1866 static
1867 HICON
CURSORICON_CopyImage(_In_ HICON hicon,_In_ BOOL bIcon,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuFlags)1868 CURSORICON_CopyImage(
1869 _In_ HICON hicon,
1870 _In_ BOOL bIcon,
1871 _In_ int cxDesired,
1872 _In_ int cyDesired,
1873 _In_ UINT fuFlags
1874 )
1875 {
1876 HICON ret = NULL;
1877 ICONINFO ii;
1878 CURSORDATA CursorData;
1879
1880 if (fuFlags & LR_COPYFROMRESOURCE)
1881 {
1882 /* Get the icon module/resource names */
1883 UNICODE_STRING ustrModule;
1884 UNICODE_STRING ustrRsrc;
1885 HMODULE hModule;
1886
1887 ustrModule.MaximumLength = 0;
1888 ustrRsrc.MaximumLength = 0;
1889
1890 /* Get the buffer size */
1891 if (!NtUserGetIconInfo(hicon, NULL, &ustrModule, &ustrRsrc, NULL, FALSE))
1892 {
1893 return NULL;
1894 }
1895
1896 ustrModule.Buffer = HeapAlloc(GetProcessHeap(), 0, ustrModule.MaximumLength);
1897 if (!ustrModule.Buffer)
1898 {
1899 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1900 return NULL;
1901 }
1902
1903 if (ustrRsrc.MaximumLength)
1904 {
1905 ustrRsrc.Buffer = HeapAlloc(GetProcessHeap(), 0, ustrRsrc.MaximumLength);
1906 if (!ustrRsrc.Buffer)
1907 {
1908 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1909 SetLastError(ERROR_NOT_ENOUGH_MEMORY);
1910 return NULL;
1911 }
1912 }
1913
1914 if (!NtUserGetIconInfo(hicon, NULL, &ustrModule, &ustrRsrc, NULL, FALSE))
1915 {
1916 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1917 if (!IS_INTRESOURCE(ustrRsrc.Buffer))
1918 HeapFree(GetProcessHeap(), 0, ustrRsrc.Buffer);
1919 return NULL;
1920 }
1921
1922 /* NULL-terminate our strings */
1923 ustrModule.Buffer[ustrModule.Length/sizeof(WCHAR)] = UNICODE_NULL;
1924 if (!IS_INTRESOURCE(ustrRsrc.Buffer))
1925 ustrRsrc.Buffer[ustrRsrc.Length/sizeof(WCHAR)] = UNICODE_NULL;
1926
1927 TRACE("Got module %wZ, resource %p (%S).\n", &ustrModule,
1928 ustrRsrc.Buffer, IS_INTRESOURCE(ustrRsrc.Buffer) ? L"" : ustrRsrc.Buffer);
1929
1930 /* Get the module handle or load the module */
1931 hModule = LoadLibraryExW(ustrModule.Buffer, NULL, /* NT6+: LOAD_LIBRARY_AS_IMAGE_RESOURCE | */ LOAD_LIBRARY_AS_DATAFILE);
1932 if (!hModule)
1933 {
1934 DWORD err = GetLastError();
1935 ERR("Unable to load/use module '%wZ' in process %lu, error %lu.\n", &ustrModule, GetCurrentProcessId(), err);
1936 SetLastError(ERROR_INVALID_PARAMETER);
1937 goto leave;
1938 }
1939
1940 /* Call the relevant function */
1941 ret = CURSORICON_LoadImageW(
1942 hModule,
1943 ustrRsrc.Buffer,
1944 cxDesired,
1945 cyDesired,
1946 fuFlags & (LR_DEFAULTSIZE | LR_SHARED),
1947 bIcon);
1948
1949 FreeLibrary(hModule);
1950
1951 /* If we're here, that means that the passed icon is shared. Don't destroy it, even if LR_COPYDELETEORG is specified */
1952 leave:
1953 HeapFree(GetProcessHeap(), 0, ustrModule.Buffer);
1954 if (!IS_INTRESOURCE(ustrRsrc.Buffer))
1955 HeapFree(GetProcessHeap(), 0, ustrRsrc.Buffer);
1956
1957 TRACE("Returning 0x%08x.\n", ret);
1958
1959 return ret;
1960 }
1961
1962 /* This is a regular copy */
1963 if (fuFlags & ~(LR_COPYDELETEORG | LR_SHARED))
1964 FIXME("Unimplemented flags: 0x%08x\n", fuFlags);
1965
1966 if (!GetIconInfo(hicon, &ii))
1967 {
1968 ERR("GetIconInfo failed.\n");
1969 return NULL;
1970 }
1971
1972 /* This is CreateIconIndirect with the LR_SHARED coat added */
1973 if (!CURSORICON_GetCursorDataFromIconInfo(&CursorData, &ii))
1974 goto Leave;
1975
1976 if (fuFlags & LR_SHARED)
1977 CursorData.CURSORF_flags |= CURSORF_LRSHARED;
1978
1979 ret = NtUserxCreateEmptyCurObject(FALSE);
1980 if (!ret)
1981 goto Leave;
1982
1983 if (!NtUserSetCursorIconData(ret, NULL, NULL, &CursorData))
1984 {
1985 NtUserDestroyCursor(ret, TRUE);
1986 goto Leave;
1987 }
1988
1989 Leave:
1990 DeleteObject(ii.hbmMask);
1991 if (ii.hbmColor) DeleteObject(ii.hbmColor);
1992
1993 if (ret && (fuFlags & LR_COPYDELETEORG))
1994 DestroyIcon(hicon);
1995
1996 return ret;
1997 }
1998
1999 NTSTATUS WINAPI
User32CallCopyImageFromKernel(PVOID Arguments,ULONG ArgumentLength)2000 User32CallCopyImageFromKernel(PVOID Arguments, ULONG ArgumentLength)
2001 {
2002 PCOPYIMAGE_CALLBACK_ARGUMENTS Common;
2003 HANDLE Result;
2004 Common = (PCOPYIMAGE_CALLBACK_ARGUMENTS) Arguments;
2005
2006 Result = CopyImage(Common->hImage,
2007 Common->uType,
2008 Common->cxDesired,
2009 Common->cyDesired,
2010 Common->fuFlags);
2011
2012 return ZwCallbackReturn(&Result, sizeof(HANDLE), STATUS_SUCCESS);
2013 }
2014
2015
2016 /************* PUBLIC FUNCTIONS *******************/
2017
2018 #define COPYIMAGE_VALID_FLAGS ( \
2019 LR_SHARED | LR_COPYFROMRESOURCE | LR_CREATEDIBSECTION | LR_LOADMAP3DCOLORS | 0x800 | \
2020 LR_VGACOLOR | LR_LOADREALSIZE | LR_DEFAULTSIZE | LR_LOADTRANSPARENT | LR_LOADFROMFILE | \
2021 LR_COPYDELETEORG | LR_COPYRETURNORG | LR_COLOR | LR_MONOCHROME \
2022 )
2023
CopyImage(_In_ HANDLE hImage,_In_ UINT uType,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuFlags)2024 HANDLE WINAPI CopyImage(
2025 _In_ HANDLE hImage,
2026 _In_ UINT uType,
2027 _In_ int cxDesired,
2028 _In_ int cyDesired,
2029 _In_ UINT fuFlags
2030 )
2031 {
2032 TRACE("hImage=%p, uType=%u, cxDesired=%d, cyDesired=%d, fuFlags=%x\n",
2033 hImage, uType, cxDesired, cyDesired, fuFlags);
2034
2035 if (fuFlags & ~COPYIMAGE_VALID_FLAGS)
2036 {
2037 SetLastError(ERROR_INVALID_PARAMETER);
2038 return NULL;
2039 }
2040
2041 switch(uType)
2042 {
2043 case IMAGE_BITMAP:
2044 if (!hImage)
2045 {
2046 SetLastError(ERROR_INVALID_HANDLE);
2047 break;
2048 }
2049 return BITMAP_CopyImage(hImage, cxDesired, cyDesired, fuFlags);
2050 case IMAGE_CURSOR:
2051 case IMAGE_ICON:
2052 {
2053 HANDLE handle;
2054 if (!hImage)
2055 {
2056 SetLastError(ERROR_INVALID_CURSOR_HANDLE);
2057 break;
2058 }
2059 handle = CURSORICON_CopyImage(hImage, uType == IMAGE_ICON, cxDesired, cyDesired, fuFlags);
2060 if (!handle && (fuFlags & LR_COPYFROMRESOURCE))
2061 {
2062 /* Test if the hImage is the same size as what we want by getting
2063 * its BITMAP and comparing its dimensions to the desired size. */
2064 BITMAP bm;
2065
2066 ICONINFO iconinfo = { 0 };
2067 if (!GetIconInfo(hImage, &iconinfo))
2068 {
2069 ERR("GetIconInfo Failed. hImage %p\n", hImage);
2070 return NULL;
2071 }
2072 if (!GetObject(iconinfo.hbmColor, sizeof(bm), &bm))
2073 {
2074 ERR("GetObject Failed. iconinfo %p\n", iconinfo);
2075 return NULL;
2076 }
2077
2078 DeleteObject(iconinfo.hbmMask);
2079 DeleteObject(iconinfo.hbmColor);
2080
2081 /* If the images are the same size remove LF_COPYFROMRESOURCE and try again */
2082 if (cxDesired == bm.bmWidth && cyDesired == bm.bmHeight)
2083 {
2084 handle = CURSORICON_CopyImage(hImage, uType == IMAGE_ICON, cxDesired,
2085 cyDesired, (fuFlags & ~LR_COPYFROMRESOURCE));
2086 }
2087 }
2088 return handle;
2089 }
2090 default:
2091 SetLastError(ERROR_INVALID_PARAMETER);
2092 break;
2093 }
2094 return NULL;
2095 }
2096
CopyIcon(_In_ HICON hIcon)2097 HICON WINAPI CopyIcon(
2098 _In_ HICON hIcon
2099 )
2100 {
2101 return CURSORICON_CopyImage(hIcon, FALSE, 0, 0, 0);
2102 }
2103
DrawIcon(_In_ HDC hDC,_In_ int X,_In_ int Y,_In_ HICON hIcon)2104 BOOL WINAPI DrawIcon(
2105 _In_ HDC hDC,
2106 _In_ int X,
2107 _In_ int Y,
2108 _In_ HICON hIcon
2109 )
2110 {
2111 return DrawIconEx(hDC, X, Y, hIcon, 0, 0, 0, NULL, DI_NORMAL | DI_COMPAT | DI_DEFAULTSIZE);
2112 }
2113
DrawIconEx(_In_ HDC hdc,_In_ int xLeft,_In_ int yTop,_In_ HICON hIcon,_In_ int cxWidth,_In_ int cyWidth,_In_ UINT istepIfAniCur,_In_opt_ HBRUSH hbrFlickerFreeDraw,_In_ UINT diFlags)2114 BOOL WINAPI DrawIconEx(
2115 _In_ HDC hdc,
2116 _In_ int xLeft,
2117 _In_ int yTop,
2118 _In_ HICON hIcon,
2119 _In_ int cxWidth,
2120 _In_ int cyWidth,
2121 _In_ UINT istepIfAniCur,
2122 _In_opt_ HBRUSH hbrFlickerFreeDraw,
2123 _In_ UINT diFlags
2124 )
2125 {
2126 return NtUserDrawIconEx(hdc, xLeft, yTop, hIcon, cxWidth, cyWidth,
2127 istepIfAniCur, hbrFlickerFreeDraw, diFlags,
2128 0, 0);
2129 }
2130
GetIconInfo(_In_ HICON hIcon,_Out_ PICONINFO piconinfo)2131 BOOL WINAPI GetIconInfo(
2132 _In_ HICON hIcon,
2133 _Out_ PICONINFO piconinfo
2134 )
2135 {
2136 return NtUserGetIconInfo(hIcon, piconinfo, NULL, NULL, NULL, FALSE);
2137 }
2138
DestroyIcon(_In_ HICON hIcon)2139 BOOL WINAPI DestroyIcon(
2140 _In_ HICON hIcon
2141 )
2142 {
2143 return NtUserDestroyCursor(hIcon, FALSE);
2144 }
2145
LoadIconA(_In_opt_ HINSTANCE hInstance,_In_ LPCSTR lpIconName)2146 HICON WINAPI LoadIconA(
2147 _In_opt_ HINSTANCE hInstance,
2148 _In_ LPCSTR lpIconName
2149 )
2150 {
2151 TRACE("%p, %s\n", hInstance, debugstr_a(lpIconName));
2152
2153 return LoadImageA(hInstance,
2154 lpIconName,
2155 IMAGE_ICON,
2156 0,
2157 0,
2158 LR_SHARED | LR_DEFAULTSIZE );
2159 }
2160
LoadIconW(_In_opt_ HINSTANCE hInstance,_In_ LPCWSTR lpIconName)2161 HICON WINAPI LoadIconW(
2162 _In_opt_ HINSTANCE hInstance,
2163 _In_ LPCWSTR lpIconName
2164 )
2165 {
2166 TRACE("%p, %s\n", hInstance, debugstr_w(lpIconName));
2167
2168 return LoadImageW(hInstance,
2169 lpIconName,
2170 IMAGE_ICON,
2171 0,
2172 0,
2173 LR_SHARED | LR_DEFAULTSIZE );
2174 }
2175
LoadCursorA(_In_opt_ HINSTANCE hInstance,_In_ LPCSTR lpCursorName)2176 HCURSOR WINAPI LoadCursorA(
2177 _In_opt_ HINSTANCE hInstance,
2178 _In_ LPCSTR lpCursorName
2179 )
2180 {
2181 TRACE("%p, %s\n", hInstance, debugstr_a(lpCursorName));
2182
2183 return LoadImageA(hInstance,
2184 lpCursorName,
2185 IMAGE_CURSOR,
2186 0,
2187 0,
2188 LR_SHARED | LR_DEFAULTSIZE );
2189 }
2190
LoadCursorW(_In_opt_ HINSTANCE hInstance,_In_ LPCWSTR lpCursorName)2191 HCURSOR WINAPI LoadCursorW(
2192 _In_opt_ HINSTANCE hInstance,
2193 _In_ LPCWSTR lpCursorName
2194 )
2195 {
2196 TRACE("%p, %s\n", hInstance, debugstr_w(lpCursorName));
2197
2198 return LoadImageW(hInstance,
2199 lpCursorName,
2200 IMAGE_CURSOR,
2201 0,
2202 0,
2203 LR_SHARED | LR_DEFAULTSIZE );
2204 }
2205
LoadCursorFromFileA(_In_ LPCSTR lpFileName)2206 HCURSOR WINAPI LoadCursorFromFileA(
2207 _In_ LPCSTR lpFileName
2208 )
2209 {
2210 TRACE("%s\n", debugstr_a(lpFileName));
2211
2212 return LoadImageA(NULL,
2213 lpFileName,
2214 IMAGE_CURSOR,
2215 0,
2216 0,
2217 LR_LOADFROMFILE | LR_DEFAULTSIZE );
2218 }
2219
LoadCursorFromFileW(_In_ LPCWSTR lpFileName)2220 HCURSOR WINAPI LoadCursorFromFileW(
2221 _In_ LPCWSTR lpFileName
2222 )
2223 {
2224 TRACE("%s\n", debugstr_w(lpFileName));
2225
2226 return LoadImageW(NULL,
2227 lpFileName,
2228 IMAGE_CURSOR,
2229 0,
2230 0,
2231 LR_LOADFROMFILE | LR_DEFAULTSIZE );
2232 }
2233
LoadBitmapA(_In_opt_ HINSTANCE hInstance,_In_ LPCSTR lpBitmapName)2234 HBITMAP WINAPI LoadBitmapA(
2235 _In_opt_ HINSTANCE hInstance,
2236 _In_ LPCSTR lpBitmapName
2237 )
2238 {
2239 TRACE("%p, %s\n", hInstance, debugstr_a(lpBitmapName));
2240
2241 return LoadImageA(hInstance,
2242 lpBitmapName,
2243 IMAGE_BITMAP,
2244 0,
2245 0,
2246 0);
2247 }
2248
LoadBitmapW(_In_opt_ HINSTANCE hInstance,_In_ LPCWSTR lpBitmapName)2249 HBITMAP WINAPI LoadBitmapW(
2250 _In_opt_ HINSTANCE hInstance,
2251 _In_ LPCWSTR lpBitmapName
2252 )
2253 {
2254 TRACE("%p, %s\n", hInstance, debugstr_w(lpBitmapName));
2255
2256 return LoadImageW(hInstance,
2257 lpBitmapName,
2258 IMAGE_BITMAP,
2259 0,
2260 0,
2261 0);
2262 }
2263
LoadImageA(_In_opt_ HINSTANCE hinst,_In_ LPCSTR lpszName,_In_ UINT uType,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuLoad)2264 HANDLE WINAPI LoadImageA(
2265 _In_opt_ HINSTANCE hinst,
2266 _In_ LPCSTR lpszName,
2267 _In_ UINT uType,
2268 _In_ int cxDesired,
2269 _In_ int cyDesired,
2270 _In_ UINT fuLoad
2271 )
2272 {
2273 HANDLE res;
2274 LPWSTR u_name;
2275 DWORD len;
2276
2277 if (IS_INTRESOURCE(lpszName))
2278 return LoadImageW(hinst, (LPCWSTR)lpszName, uType, cxDesired, cyDesired, fuLoad);
2279
2280 len = MultiByteToWideChar( CP_ACP, 0, lpszName, -1, NULL, 0 );
2281 u_name = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) );
2282 MultiByteToWideChar( CP_ACP, 0, lpszName, -1, u_name, len );
2283
2284 res = LoadImageW(hinst, u_name, uType, cxDesired, cyDesired, fuLoad);
2285 HeapFree(GetProcessHeap(), 0, u_name);
2286 return res;
2287 }
2288
LoadImageW(_In_opt_ HINSTANCE hinst,_In_ LPCWSTR lpszName,_In_ UINT uType,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT fuLoad)2289 HANDLE WINAPI LoadImageW(
2290 _In_opt_ HINSTANCE hinst,
2291 _In_ LPCWSTR lpszName,
2292 _In_ UINT uType,
2293 _In_ int cxDesired,
2294 _In_ int cyDesired,
2295 _In_ UINT fuLoad
2296 )
2297 {
2298 TRACE("hinst 0x%p, name %s, uType 0x%08x, cxDesired %d, cyDesired %d, fuLoad 0x%08x.\n",
2299 hinst, debugstr_w(lpszName), uType, cxDesired, cyDesired, fuLoad);
2300 /* Redirect to each implementation */
2301 switch(uType)
2302 {
2303 case IMAGE_BITMAP:
2304 return BITMAP_LoadImageW(hinst, lpszName, cxDesired, cyDesired, fuLoad);
2305 case IMAGE_CURSOR:
2306 case IMAGE_ICON:
2307 return CURSORICON_LoadImageW(hinst, lpszName, cxDesired, cyDesired, fuLoad, uType == IMAGE_ICON);
2308 default:
2309 SetLastError(ERROR_INVALID_PARAMETER);
2310 break;
2311 }
2312 return NULL;
2313 }
2314
LookupIconIdFromDirectory(_In_ PBYTE presbits,_In_ BOOL fIcon)2315 int WINAPI LookupIconIdFromDirectory(
2316 _In_ PBYTE presbits,
2317 _In_ BOOL fIcon
2318 )
2319 {
2320 return LookupIconIdFromDirectoryEx( presbits, fIcon,
2321 fIcon ? GetSystemMetrics(SM_CXICON) : GetSystemMetrics(SM_CXCURSOR),
2322 fIcon ? GetSystemMetrics(SM_CYICON) : GetSystemMetrics(SM_CYCURSOR), fIcon ? 0 : LR_MONOCHROME );
2323 }
2324
LookupIconIdFromDirectoryEx(_In_ PBYTE presbits,_In_ BOOL fIcon,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT Flags)2325 int WINAPI LookupIconIdFromDirectoryEx(
2326 _In_ PBYTE presbits,
2327 _In_ BOOL fIcon,
2328 _In_ int cxDesired,
2329 _In_ int cyDesired,
2330 _In_ UINT Flags
2331 )
2332 {
2333 WORD bppDesired;
2334 CURSORICONDIR* dir = (CURSORICONDIR*)presbits;
2335 CURSORICONDIRENTRY* entry;
2336 int i, numMatch = 0, iIndex = -1;
2337 WORD width, height, BitCount = 0;
2338 BOOL notPaletted = FALSE;
2339 ULONG bestScore = 0xFFFFFFFF, score;
2340
2341 TRACE("%p, %x, %i, %i, %x.\n", presbits, fIcon, cxDesired, cyDesired, Flags);
2342
2343 if(!(dir && !dir->idReserved && (dir->idType & 3)))
2344 {
2345 WARN("Invalid resource.\n");
2346 return 0;
2347 }
2348
2349 if(Flags & LR_MONOCHROME)
2350 bppDesired = 1;
2351 else
2352 {
2353 HDC icScreen;
2354 icScreen = CreateICW(DISPLAYW, NULL, NULL, NULL);
2355 if(!icScreen)
2356 return FALSE;
2357
2358 bppDesired = GetDeviceCaps(icScreen, BITSPIXEL);
2359 DeleteDC(icScreen);
2360 }
2361
2362 if(!cxDesired)
2363 cxDesired = Flags & LR_DEFAULTSIZE ? GetSystemMetrics(fIcon ? SM_CXICON : SM_CXCURSOR) : 256;
2364 if(!cyDesired)
2365 cyDesired = Flags & LR_DEFAULTSIZE ? GetSystemMetrics(fIcon ? SM_CYICON : SM_CYCURSOR) : 256;
2366
2367 /* Find the best match for the desired size */
2368 for(i = 0; i < dir->idCount; i++)
2369 {
2370 entry = &dir->idEntries[i];
2371 width = fIcon ? entry->ResInfo.icon.bWidth : entry->ResInfo.cursor.wWidth;
2372 /* Height is twice as big in cursor resources */
2373 height = fIcon ? entry->ResInfo.icon.bHeight : entry->ResInfo.cursor.wHeight/2;
2374 /* 0 represents 256 */
2375 if(!width) width = 256;
2376 if(!height) height = 256;
2377 /* Calculate the "score" (lower is better) */
2378 score = 2*(abs(width - cxDesired) + abs(height - cyDesired));
2379 if( score > bestScore)
2380 continue;
2381 /* Bigger than requested lowers the score */
2382 if(width > cxDesired)
2383 score -= width - cxDesired;
2384 if(height > cyDesired)
2385 score -= height - cyDesired;
2386 if(score > bestScore)
2387 continue;
2388 if(score == bestScore)
2389 {
2390 if(entry->wBitCount > BitCount)
2391 BitCount = entry->wBitCount;
2392 numMatch++;
2393 continue;
2394 }
2395 iIndex = i;
2396 numMatch = 1;
2397 bestScore = score;
2398 BitCount = entry->wBitCount;
2399 }
2400
2401 if(numMatch == 1)
2402 {
2403 /* Only one entry fits the asked dimensions */
2404 return dir->idEntries[iIndex].wResId;
2405 }
2406
2407 /* Avoid paletted icons on non-paletted device */
2408 if (bppDesired > 8 && BitCount > 8)
2409 notPaletted = TRUE;
2410
2411 BitCount = 0;
2412 iIndex = -1;
2413 /* Now find the entry with the best depth */
2414 for(i = 0; i < dir->idCount; i++)
2415 {
2416 entry = &dir->idEntries[i];
2417 width = fIcon ? entry->ResInfo.icon.bWidth : entry->ResInfo.cursor.wWidth;
2418 height = fIcon ? entry->ResInfo.icon.bHeight : entry->ResInfo.cursor.wHeight/2;
2419 /* 0 represents 256 */
2420 if(!width) width = 256;
2421 if(!height) height = 256;
2422 /* Check if this is the best match we had */
2423 score = 2*(abs(width - cxDesired) + abs(height - cyDesired));
2424 if(width > cxDesired)
2425 score -= width - cxDesired;
2426 if(height > cyDesired)
2427 score -= height - cyDesired;
2428 if(score != bestScore)
2429 continue;
2430 /* Exact match? */
2431 if(entry->wBitCount == bppDesired)
2432 return entry->wResId;
2433 /* We take the highest possible but smaller than the display depth */
2434 if((entry->wBitCount > BitCount) && (entry->wBitCount < bppDesired))
2435 {
2436 /* Avoid paletted icons on non paletted devices */
2437 if ((entry->wBitCount <= 8) && notPaletted)
2438 continue;
2439 iIndex = i;
2440 BitCount = entry->wBitCount;
2441 }
2442 }
2443
2444 if(iIndex >= 0)
2445 return dir->idEntries[iIndex].wResId;
2446
2447 /* No inferior or equal depth available. Get the smallest bigger one */
2448 BitCount = 0xFFFF;
2449 iIndex = -1;
2450 for(i = 0; i < dir->idCount; i++)
2451 {
2452 entry = &dir->idEntries[i];
2453 width = fIcon ? entry->ResInfo.icon.bWidth : entry->ResInfo.cursor.wWidth;
2454 height = fIcon ? entry->ResInfo.icon.bHeight : entry->ResInfo.cursor.wHeight/2;
2455 /* 0 represents 256 */
2456 if(!width) width = 256;
2457 if(!height) height = 256;
2458 /* Check if this is the best match we had */
2459 score = 2*(abs(width - cxDesired) + abs(height - cyDesired));
2460 if(width > cxDesired)
2461 score -= width - cxDesired;
2462 if(height > cyDesired)
2463 score -= height - cyDesired;
2464 if(score != bestScore)
2465 continue;
2466 /* Check the bit depth */
2467 if(entry->wBitCount < BitCount)
2468 {
2469 if((entry->wBitCount <= 8) && notPaletted)
2470 continue;
2471 iIndex = i;
2472 BitCount = entry->wBitCount;
2473 }
2474 }
2475 if (iIndex >= 0)
2476 return dir->idEntries[iIndex].wResId;
2477
2478 return 0;
2479 }
2480
CreateIcon(_In_opt_ HINSTANCE hInstance,_In_ int nWidth,_In_ int nHeight,_In_ BYTE cPlanes,_In_ BYTE cBitsPixel,_In_ const BYTE * lpbANDbits,_In_ const BYTE * lpbXORbits)2481 HICON WINAPI CreateIcon(
2482 _In_opt_ HINSTANCE hInstance,
2483 _In_ int nWidth,
2484 _In_ int nHeight,
2485 _In_ BYTE cPlanes,
2486 _In_ BYTE cBitsPixel,
2487 _In_ const BYTE *lpbANDbits,
2488 _In_ const BYTE *lpbXORbits
2489 )
2490 {
2491 ICONINFO iinfo;
2492 HICON hIcon;
2493
2494 TRACE_(icon)("%dx%d, planes %d, bpp %d, xor %p, and %p\n",
2495 nWidth, nHeight, cPlanes, cBitsPixel, lpbXORbits, lpbANDbits);
2496
2497 iinfo.fIcon = TRUE;
2498 iinfo.xHotspot = nWidth / 2;
2499 iinfo.yHotspot = nHeight / 2;
2500 if (cPlanes * cBitsPixel > 1)
2501 {
2502 iinfo.hbmColor = CreateBitmap( nWidth, nHeight, cPlanes, cBitsPixel, lpbXORbits );
2503 iinfo.hbmMask = CreateBitmap( nWidth, nHeight, 1, 1, lpbANDbits );
2504 }
2505 else
2506 {
2507 iinfo.hbmMask = CreateBitmap( nWidth, nHeight * 2, 1, 1, lpbANDbits );
2508 iinfo.hbmColor = NULL;
2509 }
2510
2511 hIcon = CreateIconIndirect( &iinfo );
2512
2513 DeleteObject( iinfo.hbmMask );
2514 if (iinfo.hbmColor) DeleteObject( iinfo.hbmColor );
2515
2516 return hIcon;
2517 }
2518
CreateIconFromResource(_In_ PBYTE presbits,_In_ DWORD dwResSize,_In_ BOOL fIcon,_In_ DWORD dwVer)2519 HICON WINAPI CreateIconFromResource(
2520 _In_ PBYTE presbits,
2521 _In_ DWORD dwResSize,
2522 _In_ BOOL fIcon,
2523 _In_ DWORD dwVer
2524 )
2525 {
2526 return CreateIconFromResourceEx( presbits, dwResSize, fIcon, dwVer, 0, 0, LR_DEFAULTSIZE | LR_SHARED);
2527 }
2528
CreateIconFromResourceEx(_In_ PBYTE pbIconBits,_In_ DWORD cbIconBits,_In_ BOOL fIcon,_In_ DWORD dwVersion,_In_ int cxDesired,_In_ int cyDesired,_In_ UINT uFlags)2529 HICON WINAPI CreateIconFromResourceEx(
2530 _In_ PBYTE pbIconBits,
2531 _In_ DWORD cbIconBits,
2532 _In_ BOOL fIcon,
2533 _In_ DWORD dwVersion,
2534 _In_ int cxDesired,
2535 _In_ int cyDesired,
2536 _In_ UINT uFlags
2537 )
2538 {
2539 CURSORDATA cursorData;
2540 HICON hIcon;
2541 BOOL isAnimated;
2542
2543 TRACE("%p, %lu, %lu, %lu, %i, %i, %lu.\n", pbIconBits, cbIconBits, fIcon, dwVersion, cxDesired, cyDesired, uFlags);
2544
2545 if(uFlags & LR_DEFAULTSIZE)
2546 {
2547 if(!cxDesired) cxDesired = GetSystemMetrics(fIcon ? SM_CXICON : SM_CXCURSOR);
2548 if(!cyDesired) cyDesired = GetSystemMetrics(fIcon ? SM_CYICON : SM_CYCURSOR);
2549 }
2550
2551 ZeroMemory(&cursorData, sizeof(cursorData));
2552 cursorData.cx = cxDesired;
2553 cursorData.cy = cyDesired;
2554 cursorData.rt = (USHORT)((ULONG_PTR)(fIcon ? RT_ICON : RT_CURSOR));
2555
2556 /* Convert to win32k-ready data */
2557 if(!memcmp(pbIconBits, "RIFF", 4))
2558 {
2559 if(!CURSORICON_GetCursorDataFromANI(&cursorData, pbIconBits, cbIconBits, uFlags))
2560 {
2561 ERR("Could not get cursor data from .ani.\n");
2562 return NULL;
2563 }
2564 isAnimated = !!(cursorData.CURSORF_flags & CURSORF_ACON);
2565 }
2566 else
2567 {
2568 /* It is possible to pass Icon Directories to this API */
2569 int wResId = LookupIconIdFromDirectoryEx(pbIconBits, fIcon, cxDesired, cyDesired, uFlags);
2570 HANDLE ResHandle = NULL;
2571 #ifdef __REACTOS__
2572 if (wResId && (pbIconBits[4] != sizeof(BITMAPINFOHEADER)))
2573 #else
2574 if(wResId)
2575 #endif
2576 {
2577 HINSTANCE hinst;
2578 HRSRC hrsrc;
2579 CURSORICONDIR* pCurIconDir = (CURSORICONDIR*)pbIconBits;
2580
2581 TRACE("Pointer points to a directory structure.\n");
2582
2583 /* So this is a pointer to an icon directory structure. Find the module */
2584 if (!GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
2585 (LPCWSTR)pbIconBits,
2586 &hinst))
2587 {
2588 return NULL;
2589 }
2590
2591 /* Check we were given the right type of resource */
2592 if((fIcon && pCurIconDir->idType == 2) || (!fIcon && pCurIconDir->idType == 1))
2593 {
2594 WARN("Got a %s directory pointer, but called for a %s\n", fIcon ? "cursor" : "icon", fIcon ? "icon" : "cursor");
2595 return NULL;
2596 }
2597
2598 /* Get the relevant resource pointer */
2599 hrsrc = FindResourceW(
2600 hinst,
2601 MAKEINTRESOURCEW(wResId),
2602 fIcon ? RT_ICON : RT_CURSOR);
2603 if (!hrsrc)
2604 return NULL;
2605
2606 ResHandle = LoadResource(hinst, hrsrc);
2607 if (!ResHandle)
2608 return NULL;
2609
2610 pbIconBits = LockResource(ResHandle);
2611 if (!pbIconBits)
2612 {
2613 FreeResource(ResHandle);
2614 return NULL;
2615 }
2616 }
2617 if(!fIcon)
2618 {
2619 WORD* pt = (WORD*)pbIconBits;
2620 cursorData.xHotspot = *pt++;
2621 cursorData.yHotspot = *pt++;
2622 pbIconBits = (PBYTE)pt;
2623 }
2624
2625 if (!CURSORICON_GetCursorDataFromBMI(&cursorData, (BITMAPINFO*)pbIconBits))
2626 {
2627 ERR("Couldn't fill the CURSORDATA structure.\n");
2628 if (ResHandle)
2629 FreeResource(ResHandle);
2630 return NULL;
2631 }
2632 if (ResHandle)
2633 FreeResource(ResHandle);
2634 isAnimated = FALSE;
2635 }
2636
2637 if (uFlags & LR_SHARED)
2638 cursorData.CURSORF_flags |= CURSORF_LRSHARED;
2639
2640 hIcon = NtUserxCreateEmptyCurObject(isAnimated);
2641 if (!hIcon)
2642 goto end_error;
2643
2644 if(!NtUserSetCursorIconData(hIcon, NULL, NULL, &cursorData))
2645 {
2646 ERR("NtUserSetCursorIconData failed.\n");
2647 NtUserDestroyCursor(hIcon, TRUE);
2648 goto end_error;
2649 }
2650
2651 if(isAnimated)
2652 HeapFree(GetProcessHeap(), 0, cursorData.aspcur);
2653
2654 return hIcon;
2655
2656 /* Clean up */
2657 end_error:
2658 if(isAnimated)
2659 HeapFree(GetProcessHeap(), 0, cursorData.aspcur);
2660 DeleteObject(cursorData.hbmMask);
2661 if(cursorData.hbmColor) DeleteObject(cursorData.hbmColor);
2662 if(cursorData.hbmAlpha) DeleteObject(cursorData.hbmAlpha);
2663
2664 return NULL;
2665 }
2666
CreateIconIndirect(_In_ PICONINFO piconinfo)2667 HICON WINAPI CreateIconIndirect(
2668 _In_ PICONINFO piconinfo
2669 )
2670 {
2671 /* As simple as creating a handle, and let win32k deal with the bitmaps */
2672 HICON hiconRet;
2673 CURSORDATA cursorData;
2674
2675 TRACE("%p.\n", piconinfo);
2676
2677 ZeroMemory(&cursorData, sizeof(cursorData));
2678
2679 if(!CURSORICON_GetCursorDataFromIconInfo(&cursorData, piconinfo))
2680 return NULL;
2681
2682 hiconRet = NtUserxCreateEmptyCurObject(FALSE);
2683 if(!hiconRet)
2684 goto end_error;
2685
2686 if(!NtUserSetCursorIconData(hiconRet, NULL, NULL, &cursorData))
2687 {
2688 NtUserDestroyCursor(hiconRet, FALSE);
2689 goto end_error;
2690 }
2691
2692 TRACE("Returning 0x%08x.\n", hiconRet);
2693
2694 return hiconRet;
2695
2696 end_error:
2697 /* Clean up */
2698 DeleteObject(cursorData.hbmMask);
2699 if(cursorData.hbmColor) DeleteObject(cursorData.hbmColor);
2700 if(cursorData.hbmAlpha) DeleteObject(cursorData.hbmAlpha);
2701
2702 return NULL;
2703 }
2704
CreateCursor(_In_opt_ HINSTANCE hInst,_In_ int xHotSpot,_In_ int yHotSpot,_In_ int nWidth,_In_ int nHeight,_In_ const VOID * pvANDPlane,_In_ const VOID * pvXORPlane)2705 HCURSOR WINAPI CreateCursor(
2706 _In_opt_ HINSTANCE hInst,
2707 _In_ int xHotSpot,
2708 _In_ int yHotSpot,
2709 _In_ int nWidth,
2710 _In_ int nHeight,
2711 _In_ const VOID *pvANDPlane,
2712 _In_ const VOID *pvXORPlane
2713 )
2714 {
2715 ICONINFO info;
2716 HCURSOR hCursor;
2717
2718 TRACE_(cursor)("%dx%d spot=%d,%d xor=%p and=%p\n",
2719 nWidth, nHeight, xHotSpot, yHotSpot, pvXORPlane, pvANDPlane);
2720
2721 info.fIcon = FALSE;
2722 info.xHotspot = xHotSpot;
2723 info.yHotspot = yHotSpot;
2724 info.hbmMask = CreateBitmap( nWidth, nHeight, 1, 1, pvANDPlane );
2725 info.hbmColor = CreateBitmap( nWidth, nHeight, 1, 1, pvXORPlane );
2726 hCursor = CreateIconIndirect( &info );
2727 DeleteObject( info.hbmMask );
2728 DeleteObject( info.hbmColor );
2729 return hCursor;
2730 }
2731
SetSystemCursor(_In_ HCURSOR hcur,_In_ DWORD id)2732 BOOL WINAPI SetSystemCursor(
2733 _In_ HCURSOR hcur,
2734 _In_ DWORD id
2735 )
2736 {
2737 if (hcur == NULL)
2738 {
2739 hcur = LoadImageW(NULL, MAKEINTRESOURCEW(id), IMAGE_CURSOR, 0, 0, LR_DEFAULTSIZE);
2740 if (hcur == NULL)
2741 {
2742 return FALSE;
2743 }
2744 }
2745 return NtUserSetSystemCursor(hcur,id);
2746 }
2747
SetCursorPos(_In_ int X,_In_ int Y)2748 BOOL WINAPI SetCursorPos(
2749 _In_ int X,
2750 _In_ int Y
2751 )
2752 {
2753 return NtUserxSetCursorPos(X,Y);
2754 }
2755
GetCursorPos(_Out_ LPPOINT lpPoint)2756 BOOL WINAPI GetCursorPos(
2757 _Out_ LPPOINT lpPoint
2758 )
2759 {
2760 return NtUserxGetCursorPos(lpPoint);
2761 }
2762
ShowCursor(_In_ BOOL bShow)2763 int WINAPI ShowCursor(
2764 _In_ BOOL bShow
2765 )
2766 {
2767 return NtUserxShowCursor(bShow);
2768 }
2769
GetCursor(void)2770 HCURSOR WINAPI GetCursor(void)
2771 {
2772 return (HCURSOR)NtUserGetThreadState(THREADSTATE_GETCURSOR);
2773 }
2774
DestroyCursor(_In_ HCURSOR hCursor)2775 BOOL WINAPI DestroyCursor(
2776 _In_ HCURSOR hCursor
2777 )
2778 {
2779 return NtUserDestroyCursor(hCursor, FALSE);
2780 }
2781
2782 HCURSOR
2783 WINAPI
GetCursorFrameInfo(HCURSOR hCursor,DWORD reserved,DWORD istep,PINT rate_jiffies,DWORD * num_steps)2784 GetCursorFrameInfo(HCURSOR hCursor, DWORD reserved, DWORD istep, PINT rate_jiffies, DWORD *num_steps)
2785 {
2786 return NtUserGetCursorFrameInfo(hCursor, istep, rate_jiffies, num_steps);
2787 }
2788