Commit 8ce4c07f authored by Jacek Caban's avatar Jacek Caban Committed by Alexandre Julliard

user32: Factor out create_icon_frame.

parent 064838c5
...@@ -173,6 +173,13 @@ static UINT get_icon_steps( struct cursoricon_object *obj ) ...@@ -173,6 +173,13 @@ static UINT get_icon_steps( struct cursoricon_object *obj )
return obj->is_ani ? obj->ani.num_steps : 1; return obj->is_ani ? obj->ani.num_steps : 1;
} }
static void free_icon_frame( struct cursoricon_frame *frame )
{
if (frame->color) DeleteObject( frame->color );
if (frame->alpha) DeleteObject( frame->alpha );
if (frame->mask) DeleteObject( frame->mask );
}
static BOOL free_icon_handle( HICON handle ) static BOOL free_icon_handle( HICON handle )
{ {
struct cursoricon_object *obj = free_user_handle( handle, NTUSER_OBJ_ICON ); struct cursoricon_object *obj = free_user_handle( handle, NTUSER_OBJ_ICON );
...@@ -187,12 +194,7 @@ static BOOL free_icon_handle( HICON handle ) ...@@ -187,12 +194,7 @@ static BOOL free_icon_handle( HICON handle )
if (!obj->is_ani) if (!obj->is_ani)
{ {
struct cursoricon_frame *frame = get_icon_frame( obj, 0 ); free_icon_frame( &obj->frame );
if (frame->alpha) DeleteObject( frame->alpha );
if (frame->color) DeleteObject( frame->color );
DeleteObject( frame->mask );
release_icon_frame( obj, frame );
} }
else else
{ {
...@@ -984,54 +986,42 @@ done: ...@@ -984,54 +986,42 @@ done:
return alpha; return alpha;
} }
static BOOL create_icon_frame( const BITMAPINFO *bmi, DWORD maxsize, POINT hotspot, BOOL is_icon,
/*********************************************************************** INT width, INT height, UINT flags, struct cursoricon_frame *frame )
* create_icon_from_bmi
*
* Create an icon from its BITMAPINFO.
*/
static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE module, LPCWSTR resname,
HRSRC rsrc, POINT hotspot, BOOL bIcon, INT width, INT height,
UINT cFlag )
{ {
DWORD size, color_size, mask_size; DWORD size, color_size, mask_size, compr;
HBITMAP color = 0, mask = 0, alpha = 0;
const void *color_bits, *mask_bits; const void *color_bits, *mask_bits;
void *alpha_mask_bits = NULL; void *alpha_mask_bits = NULL;
LONG bmi_width, bmi_height;
BITMAPINFO *bmi_copy; BITMAPINFO *bmi_copy;
BOOL ret = FALSE;
BOOL do_stretch; BOOL do_stretch;
HICON hObj = 0;
HDC hdc = 0; HDC hdc = 0;
LONG bmi_width, bmi_height;
WORD bpp; WORD bpp;
DWORD compr; BOOL ret = FALSE;
memset( frame, 0, sizeof(*frame) );
/* Check bitmap header */ /* Check bitmap header */
if (bmi->bmiHeader.biSize == PNG_SIGN) if (bmi->bmiHeader.biSize == PNG_SIGN)
{ {
BITMAPINFO *bmi_png = load_png( (const char *)bmi, &maxsize ); BITMAPINFO *bmi_png;
if (bmi_png) if (!(bmi_png = load_png( (const char *)bmi, &maxsize ))) return FALSE;
{ ret = create_icon_frame( bmi_png, maxsize, hotspot, is_icon, width, height, flags, frame );
hObj = create_icon_from_bmi( bmi_png, maxsize, module, resname, HeapFree( GetProcessHeap(), 0, bmi_png );
rsrc, hotspot, bIcon, width, height, cFlag ); return ret;
HeapFree( GetProcessHeap(), 0, bmi_png );
return hObj;
}
return 0;
} }
if (maxsize < sizeof(BITMAPCOREHEADER)) if (maxsize < sizeof(BITMAPCOREHEADER))
{ {
WARN( "invalid size %u\n", maxsize ); WARN( "invalid size %u\n", maxsize );
return 0; return FALSE;
} }
if (maxsize < bmi->bmiHeader.biSize) if (maxsize < bmi->bmiHeader.biSize)
{ {
WARN( "invalid header size %u\n", bmi->bmiHeader.biSize ); WARN( "invalid header size %u\n", bmi->bmiHeader.biSize );
return 0; return FALSE;
} }
if ( (bmi->bmiHeader.biSize != sizeof(BITMAPCOREHEADER)) && if ( (bmi->bmiHeader.biSize != sizeof(BITMAPCOREHEADER)) &&
(bmi->bmiHeader.biSize != sizeof(BITMAPINFOHEADER) || (bmi->bmiHeader.biSize != sizeof(BITMAPINFOHEADER) ||
...@@ -1039,7 +1029,7 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE ...@@ -1039,7 +1029,7 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE
bmi->bmiHeader.biCompression != BI_BITFIELDS)) ) bmi->bmiHeader.biCompression != BI_BITFIELDS)) )
{ {
WARN( "invalid bitmap header %u\n", bmi->bmiHeader.biSize ); WARN( "invalid bitmap header %u\n", bmi->bmiHeader.biSize );
return 0; return FALSE;
} }
size = bitmap_info_size( bmi, DIB_RGB_COLORS ); size = bitmap_info_size( bmi, DIB_RGB_COLORS );
...@@ -1054,21 +1044,20 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE ...@@ -1054,21 +1044,20 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE
} }
if (mask_size > maxsize - size - color_size) mask_size = 0; /* no mask */ if (mask_size > maxsize - size - color_size) mask_size = 0; /* no mask */
if (cFlag & LR_DEFAULTSIZE) if (flags & LR_DEFAULTSIZE)
{ {
if (!width) width = GetSystemMetrics( bIcon ? SM_CXICON : SM_CXCURSOR ); if (!width) width = GetSystemMetrics( is_icon ? SM_CXICON : SM_CXCURSOR );
if (!height) height = GetSystemMetrics( bIcon ? SM_CYICON : SM_CYCURSOR ); if (!height) height = GetSystemMetrics( is_icon ? SM_CYICON : SM_CYCURSOR );
} }
else else
{ {
if (!width) width = bmi_width; if (!width) width = bmi_width;
if (!height) height = bmi_height/2; if (!height) height = bmi_height/2;
} }
do_stretch = (bmi_height/2 != height) || do_stretch = (bmi_height/2 != height) || (bmi_width != width);
(bmi_width != width);
/* Scale the hotspot */ /* Scale the hotspot */
if (bIcon) if (is_icon)
{ {
hotspot.x = width / 2; hotspot.x = width / 2;
hotspot.y = height / 2; hotspot.y = height / 2;
...@@ -1093,34 +1082,28 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE ...@@ -1093,34 +1082,28 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE
color_bits = (const char*)bmi + size; color_bits = (const char*)bmi + size;
mask_bits = (const char*)color_bits + color_size; mask_bits = (const char*)color_bits + color_size;
alpha = 0;
if (is_dib_monochrome( bmi )) if (is_dib_monochrome( bmi ))
{ {
if (!(mask = CreateBitmap( width, height * 2, 1, 1, NULL ))) goto done; if (!(frame->mask = CreateBitmap( width, height * 2, 1, 1, NULL ))) goto done;
color = 0;
/* copy color data into second half of mask bitmap */ /* copy color data into second half of mask bitmap */
SelectObject( hdc, mask ); SelectObject( hdc, frame->mask );
StretchDIBits( hdc, 0, height, width, height, StretchDIBits( hdc, 0, height, width, height,
0, 0, bmi_width, bmi_height, 0, 0, bmi_width, bmi_height,
color_bits, bmi_copy, DIB_RGB_COLORS, SRCCOPY ); color_bits, bmi_copy, DIB_RGB_COLORS, SRCCOPY );
} }
else else
{ {
if (!(mask = CreateBitmap( width, height, 1, 1, NULL ))) goto done; if (!(frame->mask = CreateBitmap( width, height, 1, 1, NULL ))) goto done;
if (!(color = create_color_bitmap( width, height ))) if (!(frame->color = create_color_bitmap( width, height ))) goto done;
{ SelectObject( hdc, frame->color );
DeleteObject( mask );
goto done;
}
SelectObject( hdc, color );
StretchDIBits( hdc, 0, 0, width, height, StretchDIBits( hdc, 0, 0, width, height,
0, 0, bmi_width, bmi_height, 0, 0, bmi_width, bmi_height,
color_bits, bmi_copy, DIB_RGB_COLORS, SRCCOPY ); color_bits, bmi_copy, DIB_RGB_COLORS, SRCCOPY );
if (bmi_has_alpha( bmi_copy, color_bits )) if (bmi_has_alpha( bmi_copy, color_bits ))
{ {
alpha = create_alpha_bitmap( color, bmi_copy, color_bits ); frame->alpha = create_alpha_bitmap( frame->color, bmi_copy, color_bits );
if (!mask_size) /* generate mask from alpha */ if (!mask_size) /* generate mask from alpha */
{ {
LONG x, y, dst_stride = ((bmi_width + 31) / 8) & ~3; LONG x, y, dst_stride = ((bmi_width + 31) / 8) & ~3;
...@@ -1164,35 +1147,48 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE ...@@ -1164,35 +1147,48 @@ static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE
if (mask_size) if (mask_size)
{ {
SelectObject( hdc, mask ); SelectObject( hdc, frame->mask );
StretchDIBits( hdc, 0, 0, width, height, StretchDIBits( hdc, 0, 0, width, height,
0, 0, bmi_width, bmi_height, 0, 0, bmi_width, bmi_height,
mask_bits, bmi_copy, DIB_RGB_COLORS, SRCCOPY ); mask_bits, bmi_copy, DIB_RGB_COLORS, SRCCOPY );
} }
frame->delay = ~0;
frame->width = width;
frame->height = height;
frame->hotspot = hotspot;
ret = TRUE; ret = TRUE;
done: done:
if (!ret) free_icon_frame( frame );
DeleteDC( hdc ); DeleteDC( hdc );
HeapFree( GetProcessHeap(), 0, bmi_copy ); HeapFree( GetProcessHeap(), 0, bmi_copy );
HeapFree( GetProcessHeap(), 0, alpha_mask_bits ); HeapFree( GetProcessHeap(), 0, alpha_mask_bits );
return ret;
}
/***********************************************************************
* create_icon_from_bmi
*
* Create an icon from its BITMAPINFO.
*/
static HICON create_icon_from_bmi( const BITMAPINFO *bmi, DWORD maxsize, HMODULE module, LPCWSTR resname,
HRSRC rsrc, POINT hotspot, BOOL bIcon, INT width, INT height,
UINT flags )
{
struct cursoricon_frame frame;
HICON ret;
if (!create_icon_frame( bmi, maxsize, hotspot, bIcon, width, height, flags, &frame )) return 0;
ret = alloc_icon_handle( FALSE, 0 );
if (ret) if (ret)
hObj = alloc_icon_handle( FALSE, 0 );
if (hObj)
{ {
struct cursoricon_object *info = get_icon_ptr( hObj ); struct cursoricon_object *info = get_icon_ptr( ret );
struct cursoricon_frame *frame;
info->is_icon = bIcon; info->is_icon = bIcon;
frame = get_icon_frame( info, 0 ); info->frame = frame;
frame->delay = ~0;
frame->width = width;
frame->height = height;
frame->color = color;
frame->mask = mask;
frame->alpha = alpha;
frame->hotspot = hotspot;
release_icon_frame( info, frame );
if (!IS_INTRESOURCE(resname)) if (!IS_INTRESOURCE(resname))
{ {
info->resname = HeapAlloc( GetProcessHeap(), 0, (lstrlenW(resname) + 1) * sizeof(WCHAR) ); info->resname = HeapAlloc( GetProcessHeap(), 0, (lstrlenW(resname) + 1) * sizeof(WCHAR) );
...@@ -1213,7 +1209,7 @@ done: ...@@ -1213,7 +1209,7 @@ done:
} }
} }
if (cFlag & LR_SHARED) if (flags & LR_SHARED)
{ {
info->is_shared = TRUE; info->is_shared = TRUE;
if (module) if (module)
...@@ -1226,11 +1222,9 @@ done: ...@@ -1226,11 +1222,9 @@ done:
} }
else else
{ {
DeleteObject( color ); free_icon_frame( &frame );
DeleteObject( alpha );
DeleteObject( mask );
} }
return hObj; return ret;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment