2022-04-07 13:24:55 +00:00
|
|
|
/*
|
|
|
|
* Default window procedure
|
|
|
|
*
|
|
|
|
* Copyright 1993, 1996 Alexandre Julliard
|
|
|
|
* Copyright 1995 Alex Korobka
|
|
|
|
* Copyright 2022 Jacek Caban for CodeWeavers
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
#pragma makedep unix
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "win32u_private.h"
|
|
|
|
#include "ntuser_private.h"
|
|
|
|
#include "wine/server.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(win);
|
|
|
|
|
|
|
|
|
2022-05-11 19:00:01 +00:00
|
|
|
void fill_rect( HDC dc, const RECT *rect, HBRUSH hbrush )
|
|
|
|
{
|
|
|
|
HBRUSH prev_brush;
|
|
|
|
|
2022-05-16 11:57:38 +00:00
|
|
|
if (hbrush <= (HBRUSH)(COLOR_MENUBAR + 1)) hbrush = get_sys_color_brush( HandleToULong(hbrush) - 1 );
|
|
|
|
|
2022-05-11 19:00:01 +00:00
|
|
|
prev_brush = NtGdiSelectBrush( dc, hbrush );
|
|
|
|
NtGdiPatBlt( dc, rect->left, rect->top, rect->right - rect->left, rect->bottom - rect->top, PATCOPY );
|
|
|
|
if (prev_brush) NtGdiSelectBrush( dc, prev_brush );
|
|
|
|
}
|
|
|
|
|
2022-04-15 18:43:19 +00:00
|
|
|
/***********************************************************************
|
|
|
|
* AdjustWindowRectEx (win32u.so)
|
|
|
|
*/
|
|
|
|
BOOL WINAPI AdjustWindowRectEx( RECT *rect, DWORD style, BOOL menu, DWORD ex_style )
|
|
|
|
{
|
|
|
|
NONCLIENTMETRICSW ncm;
|
|
|
|
int adjust = 0;
|
|
|
|
|
|
|
|
ncm.cbSize = sizeof(ncm);
|
|
|
|
NtUserSystemParametersInfo( SPI_GETNONCLIENTMETRICS, 0, &ncm, 0 );
|
|
|
|
|
|
|
|
if ((ex_style & (WS_EX_STATICEDGE|WS_EX_DLGMODALFRAME)) == WS_EX_STATICEDGE)
|
|
|
|
adjust = 1; /* for the outer frame always present */
|
|
|
|
else if ((ex_style & WS_EX_DLGMODALFRAME) || (style & (WS_THICKFRAME|WS_DLGFRAME)))
|
|
|
|
adjust = 2; /* outer */
|
|
|
|
|
|
|
|
if (style & WS_THICKFRAME)
|
|
|
|
adjust += ncm.iBorderWidth + ncm.iPaddedBorderWidth; /* The resize border */
|
|
|
|
|
|
|
|
if ((style & (WS_BORDER|WS_DLGFRAME)) || (ex_style & WS_EX_DLGMODALFRAME))
|
|
|
|
adjust++; /* The other border */
|
|
|
|
|
|
|
|
InflateRect( rect, adjust, adjust );
|
|
|
|
|
|
|
|
if ((style & WS_CAPTION) == WS_CAPTION)
|
|
|
|
{
|
|
|
|
if (ex_style & WS_EX_TOOLWINDOW)
|
|
|
|
rect->top -= ncm.iSmCaptionHeight + 1;
|
|
|
|
else
|
|
|
|
rect->top -= ncm.iCaptionHeight + 1;
|
|
|
|
}
|
|
|
|
if (menu) rect->top -= ncm.iMenuHeight + 1;
|
|
|
|
|
|
|
|
if (ex_style & WS_EX_CLIENTEDGE)
|
|
|
|
InflateRect( rect, get_system_metrics(SM_CXEDGE), get_system_metrics(SM_CYEDGE) );
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2022-04-07 13:24:55 +00:00
|
|
|
static BOOL set_window_text( HWND hwnd, const void *text, BOOL ansi )
|
|
|
|
{
|
|
|
|
static const WCHAR emptyW[] = { 0 };
|
|
|
|
WCHAR *str;
|
|
|
|
WND *win;
|
|
|
|
|
|
|
|
/* check for string, as static icons, bitmaps (SS_ICON, SS_BITMAP)
|
|
|
|
* may have child window IDs instead of window name */
|
|
|
|
if (text && IS_INTRESOURCE(text)) return FALSE;
|
|
|
|
|
|
|
|
if (text)
|
|
|
|
{
|
|
|
|
if (ansi) str = towstr( text );
|
|
|
|
else str = wcsdup( text );
|
|
|
|
if (!str) return FALSE;
|
|
|
|
}
|
|
|
|
else str = NULL;
|
|
|
|
|
|
|
|
TRACE( "%s\n", debugstr_w(str) );
|
|
|
|
|
|
|
|
if (!(win = get_win_ptr( hwnd )))
|
|
|
|
{
|
|
|
|
free( str );
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
free( win->text );
|
|
|
|
win->text = str;
|
|
|
|
SERVER_START_REQ( set_window_text )
|
|
|
|
{
|
|
|
|
req->handle = wine_server_user_handle( hwnd );
|
|
|
|
if (str) wine_server_add_data( req, str, lstrlenW( str ) * sizeof(WCHAR) );
|
|
|
|
wine_server_call( req );
|
|
|
|
}
|
|
|
|
SERVER_END_REQ;
|
|
|
|
|
|
|
|
release_win_ptr( win );
|
|
|
|
|
|
|
|
user_driver->pSetWindowText( hwnd, str ? str : emptyW );
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2022-05-11 18:14:50 +00:00
|
|
|
static HICON get_window_icon( HWND hwnd, WPARAM type )
|
|
|
|
{
|
|
|
|
HICON ret;
|
|
|
|
WND *win;
|
|
|
|
|
|
|
|
if (!(win = get_win_ptr( hwnd ))) return 0;
|
|
|
|
|
|
|
|
switch(type)
|
|
|
|
{
|
|
|
|
case ICON_SMALL:
|
|
|
|
ret = win->hIconSmall;
|
|
|
|
break;
|
|
|
|
case ICON_BIG:
|
|
|
|
ret = win->hIcon;
|
|
|
|
break;
|
|
|
|
case ICON_SMALL2:
|
|
|
|
ret = win->hIconSmall ? win->hIconSmall : win->hIconSmall2;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
release_win_ptr( win );
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-04-07 13:25:11 +00:00
|
|
|
static HICON set_window_icon( HWND hwnd, WPARAM type, HICON icon )
|
|
|
|
{
|
|
|
|
HICON ret = 0;
|
|
|
|
WND *win;
|
|
|
|
|
|
|
|
if (!(win = get_win_ptr( hwnd ))) return 0;
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case ICON_SMALL:
|
|
|
|
ret = win->hIconSmall;
|
2022-05-05 11:34:25 +00:00
|
|
|
if (ret && !icon && win->hIcon)
|
2022-04-07 13:25:11 +00:00
|
|
|
{
|
2022-05-05 11:34:25 +00:00
|
|
|
win->hIconSmall2 = CopyImage( win->hIcon, IMAGE_ICON,
|
|
|
|
get_system_metrics( SM_CXSMICON ),
|
|
|
|
get_system_metrics( SM_CYSMICON ), 0 );
|
2022-04-07 13:25:11 +00:00
|
|
|
}
|
|
|
|
else if (icon && win->hIconSmall2)
|
|
|
|
{
|
|
|
|
NtUserDestroyCursor( win->hIconSmall2, 0 );
|
|
|
|
win->hIconSmall2 = NULL;
|
|
|
|
}
|
|
|
|
win->hIconSmall = icon;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ICON_BIG:
|
|
|
|
ret = win->hIcon;
|
|
|
|
if (win->hIconSmall2)
|
|
|
|
{
|
|
|
|
NtUserDestroyCursor( win->hIconSmall2, 0 );
|
|
|
|
win->hIconSmall2 = NULL;
|
|
|
|
}
|
2022-05-05 11:34:25 +00:00
|
|
|
if (icon && !win->hIconSmall)
|
2022-04-07 13:25:11 +00:00
|
|
|
{
|
2022-05-05 11:34:25 +00:00
|
|
|
win->hIconSmall2 = CopyImage( icon, IMAGE_ICON,
|
|
|
|
get_system_metrics( SM_CXSMICON ),
|
|
|
|
get_system_metrics( SM_CYSMICON ), 0 );
|
2022-04-07 13:25:11 +00:00
|
|
|
}
|
|
|
|
win->hIcon = icon;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
release_win_ptr( win );
|
|
|
|
|
|
|
|
user_driver->pSetWindowIcon( hwnd, type, icon );
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-05-11 19:26:52 +00:00
|
|
|
static LONG handle_window_pos_changing( HWND hwnd, WINDOWPOS *winpos )
|
|
|
|
{
|
|
|
|
LONG style = get_window_long( hwnd, GWL_STYLE );
|
|
|
|
|
|
|
|
if (winpos->flags & SWP_NOSIZE) return 0;
|
|
|
|
if ((style & WS_THICKFRAME) || ((style & (WS_POPUP | WS_CHILD)) == 0))
|
|
|
|
{
|
|
|
|
MINMAXINFO info = get_min_max_info( hwnd );
|
|
|
|
winpos->cx = min( winpos->cx, info.ptMaxTrackSize.x );
|
|
|
|
winpos->cy = min( winpos->cy, info.ptMaxTrackSize.y );
|
|
|
|
if (!(style & WS_MINIMIZE))
|
|
|
|
{
|
|
|
|
winpos->cx = max( winpos->cx, info.ptMinTrackSize.x );
|
|
|
|
winpos->cy = max( winpos->cy, info.ptMinTrackSize.y );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-04-07 13:26:39 +00:00
|
|
|
static LRESULT handle_sys_command( HWND hwnd, WPARAM wparam, LPARAM lparam )
|
|
|
|
{
|
|
|
|
if (!is_window_enabled( hwnd )) return 0;
|
|
|
|
|
|
|
|
if (call_hooks( WH_CBT, HCBT_SYSCOMMAND, wparam, lparam, TRUE ))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!user_driver->pSysCommand( hwnd, wparam, lparam ))
|
|
|
|
return 0;
|
|
|
|
|
2022-04-15 12:54:16 +00:00
|
|
|
switch (wparam & 0xfff0)
|
|
|
|
{
|
|
|
|
case SC_MINIMIZE:
|
|
|
|
show_owned_popups( hwnd, FALSE );
|
|
|
|
NtUserShowWindow( hwnd, SW_MINIMIZE );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SC_MAXIMIZE:
|
|
|
|
if (is_iconic(hwnd)) show_owned_popups( hwnd, TRUE );
|
|
|
|
NtUserShowWindow( hwnd, SW_MAXIMIZE );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SC_RESTORE:
|
|
|
|
if (is_iconic( hwnd )) show_owned_popups( hwnd, TRUE );
|
|
|
|
NtUserShowWindow( hwnd, SW_RESTORE );
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return 1; /* handle on client side */
|
|
|
|
}
|
|
|
|
return 0;
|
2022-04-07 13:26:39 +00:00
|
|
|
}
|
2022-04-07 13:25:11 +00:00
|
|
|
|
2022-04-07 13:24:55 +00:00
|
|
|
LRESULT default_window_proc( HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam, BOOL ansi )
|
|
|
|
{
|
|
|
|
LRESULT result = 0;
|
|
|
|
|
|
|
|
switch (msg)
|
|
|
|
{
|
|
|
|
case WM_NCCREATE:
|
|
|
|
if (lparam)
|
|
|
|
{
|
|
|
|
CREATESTRUCTW *cs = (CREATESTRUCTW *)lparam;
|
|
|
|
set_window_text( hwnd, cs->lpszName, ansi );
|
|
|
|
result = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_NCDESTROY:
|
|
|
|
{
|
|
|
|
WND *win = get_win_ptr( hwnd );
|
|
|
|
if (!win) return 0;
|
|
|
|
free( win->text );
|
|
|
|
win->text = NULL;
|
|
|
|
if (user_callbacks) user_callbacks->free_win_ptr( win );
|
|
|
|
win->pScroll = NULL;
|
|
|
|
release_win_ptr( win );
|
2022-05-11 18:39:30 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-05-11 19:26:52 +00:00
|
|
|
case WM_WINDOWPOSCHANGING:
|
|
|
|
return handle_window_pos_changing( hwnd, (WINDOWPOS *)lparam );
|
|
|
|
|
2022-05-11 18:39:30 +00:00
|
|
|
case WM_PAINTICON:
|
|
|
|
case WM_PAINT:
|
|
|
|
{
|
|
|
|
PAINTSTRUCT ps;
|
|
|
|
HDC hdc = NtUserBeginPaint( hwnd, &ps );
|
|
|
|
if (hdc)
|
|
|
|
{
|
|
|
|
HICON icon;
|
|
|
|
if (is_iconic(hwnd) && ((icon = UlongToHandle( get_class_long( hwnd, GCLP_HICON, FALSE )))))
|
|
|
|
{
|
|
|
|
RECT rc;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
get_client_rect( hwnd, &rc );
|
|
|
|
x = (rc.right - rc.left - get_system_metrics( SM_CXICON )) / 2;
|
|
|
|
y = (rc.bottom - rc.top - get_system_metrics( SM_CYICON )) / 2;
|
|
|
|
TRACE( "Painting class icon: vis rect=(%s)\n", wine_dbgstr_rect(&ps.rcPaint) );
|
|
|
|
NtUserDrawIconEx( hdc, x, y, icon, 0, 0, 0, 0, DI_NORMAL | DI_COMPAT | DI_DEFAULTSIZE );
|
|
|
|
}
|
|
|
|
NtUserEndPaint( hwnd, &ps );
|
|
|
|
}
|
|
|
|
break;
|
2022-04-07 13:24:55 +00:00
|
|
|
}
|
|
|
|
|
2022-05-11 18:48:07 +00:00
|
|
|
case WM_SYNCPAINT:
|
|
|
|
NtUserRedrawWindow ( hwnd, NULL, 0, RDW_ERASENOW | RDW_ERASE | RDW_ALLCHILDREN );
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case WM_SETREDRAW:
|
|
|
|
if (wparam) set_window_style( hwnd, WS_VISIBLE, 0 );
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NtUserRedrawWindow( hwnd, NULL, 0, RDW_ALLCHILDREN | RDW_VALIDATE );
|
|
|
|
set_window_style( hwnd, 0, WS_VISIBLE );
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case WM_CLOSE:
|
|
|
|
NtUserDestroyWindow( hwnd );
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case WM_MOUSEACTIVATE:
|
|
|
|
if (get_window_long( hwnd, GWL_STYLE ) & WS_CHILD)
|
|
|
|
{
|
|
|
|
result = send_message( get_parent(hwnd), WM_MOUSEACTIVATE, wparam, lparam );
|
|
|
|
if (result) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Caption clicks are handled by NC_HandleNCLButtonDown() */
|
|
|
|
result = HIWORD(lparam) == WM_LBUTTONDOWN && LOWORD(lparam) == HTCAPTION ?
|
|
|
|
MA_NOACTIVATE : MA_ACTIVATE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_ACTIVATE:
|
|
|
|
/* The default action in Windows is to set the keyboard focus to
|
|
|
|
* the window, if it's being activated and not minimized */
|
|
|
|
if (LOWORD(wparam) != WA_INACTIVE && !is_iconic( hwnd )) NtUserSetFocus( hwnd );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case WM_MOUSEWHEEL:
|
|
|
|
if (get_window_long( hwnd, GWL_STYLE ) & WS_CHILD)
|
|
|
|
result = send_message( get_parent( hwnd ), WM_MOUSEWHEEL, wparam, lparam );
|
|
|
|
break;
|
|
|
|
|
2022-05-11 19:00:01 +00:00
|
|
|
case WM_ERASEBKGND:
|
|
|
|
case WM_ICONERASEBKGND:
|
|
|
|
{
|
|
|
|
RECT rect;
|
|
|
|
HDC hdc = (HDC)wparam;
|
|
|
|
HBRUSH hbr = UlongToHandle( get_class_long( hwnd, GCLP_HBRBACKGROUND, FALSE ));
|
|
|
|
if (!hbr) break;
|
|
|
|
|
|
|
|
if (get_class_long( hwnd, GCL_STYLE, FALSE ) & CS_PARENTDC)
|
|
|
|
{
|
|
|
|
/* can't use GetClipBox with a parent DC or we fill the whole parent */
|
|
|
|
get_client_rect( hwnd, &rect );
|
|
|
|
NtGdiTransformPoints( hdc, (POINT *)&rect, (POINT *)&rect, 1, NtGdiDPtoLP );
|
|
|
|
}
|
|
|
|
else NtGdiGetAppClipBox( hdc, &rect );
|
|
|
|
fill_rect( hdc, &rect, hbr );
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2022-05-11 18:48:07 +00:00
|
|
|
case WM_GETDLGCODE:
|
|
|
|
break;
|
|
|
|
|
2022-04-07 13:24:55 +00:00
|
|
|
case WM_SETTEXT:
|
|
|
|
result = set_window_text( hwnd, (void *)lparam, ansi );
|
|
|
|
break;
|
2022-04-07 13:25:11 +00:00
|
|
|
|
|
|
|
case WM_SETICON:
|
|
|
|
result = (LRESULT)set_window_icon( hwnd, wparam, (HICON)lparam );
|
|
|
|
break;
|
2022-04-07 13:26:39 +00:00
|
|
|
|
2022-05-11 18:14:50 +00:00
|
|
|
case WM_GETICON:
|
|
|
|
result = (LRESULT)get_window_icon( hwnd, wparam );
|
|
|
|
break;
|
|
|
|
|
2022-04-07 13:26:39 +00:00
|
|
|
case WM_SYSCOMMAND:
|
|
|
|
result = handle_sys_command( hwnd, wparam, lparam );
|
|
|
|
break;
|
2022-04-07 13:24:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2022-05-19 12:23:17 +00:00
|
|
|
|
|
|
|
LRESULT desktop_window_proc( HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam )
|
|
|
|
{
|
|
|
|
static const WCHAR wine_display_device_guidW[] =
|
|
|
|
{'_','_','w','i','n','e','_','d','i','s','p','l','a','y','_','d','e','v','i','c','e',
|
|
|
|
'_','g','u','i','d',0};
|
|
|
|
|
|
|
|
switch (msg)
|
|
|
|
{
|
|
|
|
case WM_NCCREATE:
|
|
|
|
{
|
|
|
|
CREATESTRUCTW *cs = (CREATESTRUCTW *)lparam;
|
|
|
|
const GUID *guid = cs->lpCreateParams;
|
|
|
|
|
|
|
|
if (guid)
|
|
|
|
{
|
|
|
|
ATOM atom = 0;
|
|
|
|
char buffer[37];
|
|
|
|
WCHAR bufferW[37];
|
|
|
|
|
|
|
|
if (NtUserGetAncestor( hwnd, GA_PARENT )) return FALSE; /* refuse to create non-desktop window */
|
|
|
|
|
|
|
|
sprintf( buffer, "%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
|
|
|
|
(unsigned int)guid->Data1, guid->Data2, guid->Data3,
|
|
|
|
guid->Data4[0], guid->Data4[1], guid->Data4[2], guid->Data4[3],
|
|
|
|
guid->Data4[4], guid->Data4[5], guid->Data4[6], guid->Data4[7] );
|
|
|
|
NtAddAtom( bufferW, asciiz_to_unicode( bufferW, buffer ) - sizeof(WCHAR), &atom );
|
|
|
|
NtUserSetProp( hwnd, wine_display_device_guidW, ULongToHandle( atom ) );
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
case WM_NCCALCSIZE:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return default_window_proc( hwnd, msg, wparam, lparam, FALSE );
|
|
|
|
}
|