| /* |
| * Default dialog procedure |
| * |
| * Copyright 1993, 1996 Alexandre Julliard |
| * |
| */ |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "wingdi.h" |
| #include "wine/winuser16.h" |
| #include "controls.h" |
| #include "win.h" |
| #include "winproc.h" |
| #include "debugtools.h" |
| |
| DEFAULT_DEBUG_CHANNEL(dialog); |
| |
| |
| /*********************************************************************** |
| * DEFDLG_GetDlgProc |
| */ |
| static WNDPROC DEFDLG_GetDlgProc( HWND hwnd ) |
| { |
| WNDPROC ret; |
| WND *wndPtr = WIN_GetPtr( hwnd ); |
| |
| if (!wndPtr) return 0; |
| if (wndPtr == WND_OTHER_PROCESS) |
| { |
| ERR( "cannot get dlg proc %x from other process\n", hwnd ); |
| return 0; |
| } |
| ret = *(WNDPROC *)((char *)wndPtr->wExtra + DWL_DLGPROC); |
| WIN_ReleasePtr( wndPtr ); |
| return ret; |
| } |
| |
| /*********************************************************************** |
| * DEFDLG_SetFocus |
| * |
| * Set the focus to a control of the dialog, selecting the text if |
| * the control is an edit dialog. |
| */ |
| static void DEFDLG_SetFocus( HWND hwndDlg, HWND hwndCtrl ) |
| { |
| HWND hwndPrev = GetFocus(); |
| |
| if (IsChild( hwndDlg, hwndPrev )) |
| { |
| if (SendMessageW( hwndPrev, WM_GETDLGCODE, 0, 0 ) & DLGC_HASSETSEL) |
| SendMessageW( hwndPrev, EM_SETSEL, -1, 0 ); |
| } |
| if (SendMessageW( hwndCtrl, WM_GETDLGCODE, 0, 0 ) & DLGC_HASSETSEL) |
| SendMessageW( hwndCtrl, EM_SETSEL, 0, -1 ); |
| SetFocus( hwndCtrl ); |
| } |
| |
| |
| /*********************************************************************** |
| * DEFDLG_SaveFocus |
| */ |
| static void DEFDLG_SaveFocus( HWND hwnd ) |
| { |
| DIALOGINFO *infoPtr; |
| HWND hwndFocus = GetFocus(); |
| |
| if (!hwndFocus || !IsChild( hwnd, hwndFocus )) return; |
| if (!(infoPtr = DIALOG_get_info( hwnd ))) return; |
| infoPtr->hwndFocus = hwndFocus; |
| /* Remove default button */ |
| } |
| |
| |
| /*********************************************************************** |
| * DEFDLG_RestoreFocus |
| */ |
| static void DEFDLG_RestoreFocus( HWND hwnd ) |
| { |
| DIALOGINFO *infoPtr; |
| |
| if (IsIconic( hwnd )) return; |
| if (!(infoPtr = DIALOG_get_info( hwnd ))) return; |
| if (!IsWindow( infoPtr->hwndFocus )) return; |
| /* Don't set the focus back to controls if EndDialog is already called.*/ |
| if (!(infoPtr->flags & DF_END)) |
| { |
| DEFDLG_SetFocus( hwnd, infoPtr->hwndFocus ); |
| return; |
| } |
| /* This used to set infoPtr->hwndFocus to NULL for no apparent reason, |
| sometimes losing focus when receiving WM_SETFOCUS messages. */ |
| } |
| |
| |
| /*********************************************************************** |
| * DEFDLG_FindDefButton |
| * |
| * Find the current default push-button. |
| */ |
| static HWND DEFDLG_FindDefButton( HWND hwndDlg ) |
| { |
| HWND hwndChild = GetWindow( hwndDlg, GW_CHILD ); |
| while (hwndChild) |
| { |
| if (SendMessageW( hwndChild, WM_GETDLGCODE, 0, 0 ) & DLGC_DEFPUSHBUTTON) |
| break; |
| hwndChild = GetWindow( hwndChild, GW_HWNDNEXT ); |
| } |
| return hwndChild; |
| } |
| |
| |
| /*********************************************************************** |
| * DEFDLG_SetDefButton |
| * |
| * Set the new default button to be hwndNew. |
| */ |
| static BOOL DEFDLG_SetDefButton( HWND hwndDlg, DIALOGINFO *dlgInfo, |
| HWND hwndNew ) |
| { |
| if (hwndNew && |
| !(SendMessageW(hwndNew, WM_GETDLGCODE, 0, 0 ) & DLGC_UNDEFPUSHBUTTON)) |
| return FALSE; /* Destination is not a push button */ |
| |
| if (dlgInfo->idResult) /* There's already a default pushbutton */ |
| { |
| HWND hwndOld = GetDlgItem( hwndDlg, dlgInfo->idResult ); |
| if (SendMessageA( hwndOld, WM_GETDLGCODE, 0, 0) & DLGC_DEFPUSHBUTTON) |
| SendMessageA( hwndOld, BM_SETSTYLE, BS_PUSHBUTTON, TRUE ); |
| } |
| if (hwndNew) |
| { |
| SendMessageA( hwndNew, BM_SETSTYLE, BS_DEFPUSHBUTTON, TRUE ); |
| dlgInfo->idResult = GetDlgCtrlID( hwndNew ); |
| } |
| else dlgInfo->idResult = 0; |
| return TRUE; |
| } |
| |
| |
| /*********************************************************************** |
| * DEFDLG_Proc |
| * |
| * Implementation of DefDlgProc(). Only handle messages that need special |
| * handling for dialogs. |
| */ |
| static LRESULT DEFDLG_Proc( HWND hwnd, UINT msg, WPARAM wParam, |
| LPARAM lParam, DIALOGINFO *dlgInfo ) |
| { |
| switch(msg) |
| { |
| case WM_ERASEBKGND: |
| { |
| HBRUSH brush = SendMessageW( hwnd, WM_CTLCOLORDLG, wParam, (LPARAM)hwnd ); |
| if (!brush) brush = DefWindowProcW( hwnd, WM_CTLCOLORDLG, wParam, (LPARAM)hwnd ); |
| if (brush) |
| { |
| RECT rect; |
| HDC hdc = (HDC)wParam; |
| GetClientRect( hwnd, &rect ); |
| DPtoLP( hdc, (LPPOINT)&rect, 2 ); |
| FillRect( hdc, &rect, brush ); |
| } |
| return 1; |
| } |
| case WM_NCDESTROY: |
| if ((dlgInfo = (DIALOGINFO *)SetWindowLongW( hwnd, DWL_WINE_DIALOGINFO, 0 ))) |
| { |
| /* Free dialog heap (if created) */ |
| if (dlgInfo->hDialogHeap) |
| { |
| GlobalUnlock16(dlgInfo->hDialogHeap); |
| GlobalFree16(dlgInfo->hDialogHeap); |
| } |
| if (dlgInfo->hUserFont) DeleteObject( dlgInfo->hUserFont ); |
| if (dlgInfo->hMenu) DestroyMenu( dlgInfo->hMenu ); |
| WINPROC_FreeProc( DEFDLG_GetDlgProc( hwnd ), WIN_PROC_WINDOW ); |
| HeapFree( GetProcessHeap(), 0, dlgInfo ); |
| } |
| /* Window clean-up */ |
| return DefWindowProcA( hwnd, msg, wParam, lParam ); |
| |
| case WM_SHOWWINDOW: |
| if (!wParam) DEFDLG_SaveFocus( hwnd ); |
| return DefWindowProcA( hwnd, msg, wParam, lParam ); |
| |
| case WM_ACTIVATE: |
| if (wParam) DEFDLG_RestoreFocus( hwnd ); |
| else DEFDLG_SaveFocus( hwnd ); |
| return 0; |
| |
| case WM_SETFOCUS: |
| DEFDLG_RestoreFocus( hwnd ); |
| return 0; |
| |
| case DM_SETDEFID: |
| if (dlgInfo->flags & DF_END) return 1; |
| DEFDLG_SetDefButton( hwnd, dlgInfo, |
| wParam ? GetDlgItem( hwnd, wParam ) : 0 ); |
| return 1; |
| |
| case DM_GETDEFID: |
| { |
| HWND hwndDefId; |
| if (dlgInfo->flags & DF_END) return 0; |
| if (dlgInfo->idResult) |
| return MAKELONG( dlgInfo->idResult, DC_HASDEFID ); |
| if ((hwndDefId = DEFDLG_FindDefButton( hwnd ))) |
| return MAKELONG( GetDlgCtrlID( hwndDefId ), DC_HASDEFID); |
| } |
| return 0; |
| |
| case WM_NEXTDLGCTL: |
| { |
| HWND hwndDest = (HWND)wParam; |
| if (!lParam) |
| hwndDest = GetNextDlgTabItem(hwnd, GetFocus(), wParam); |
| if (hwndDest) DEFDLG_SetFocus( hwnd, hwndDest ); |
| DEFDLG_SetDefButton( hwnd, dlgInfo, hwndDest ); |
| } |
| return 0; |
| |
| case WM_ENTERMENULOOP: |
| case WM_LBUTTONDOWN: |
| case WM_NCLBUTTONDOWN: |
| { |
| HWND hwndFocus = GetFocus(); |
| if (hwndFocus) |
| { |
| /* always make combo box hide its listbox control */ |
| if (!SendMessageA( hwndFocus, CB_SHOWDROPDOWN, FALSE, 0 )) |
| SendMessageA( GetParent(hwndFocus), CB_SHOWDROPDOWN, FALSE, 0 ); |
| } |
| } |
| return DefWindowProcA( hwnd, msg, wParam, lParam ); |
| |
| case WM_GETFONT: |
| return dlgInfo->hUserFont; |
| |
| case WM_CLOSE: |
| PostMessageA( hwnd, WM_COMMAND, IDCANCEL, |
| (LPARAM)GetDlgItem( hwnd, IDCANCEL ) ); |
| return 0; |
| |
| case WM_NOTIFYFORMAT: |
| return DefWindowProcA( hwnd, msg, wParam, lParam ); |
| } |
| return 0; |
| } |
| |
| /*********************************************************************** |
| * DEFDLG_Epilog |
| */ |
| static LRESULT DEFDLG_Epilog(HWND hwnd, UINT msg, BOOL fResult) |
| { |
| /* see SDK 3.1 */ |
| |
| if ((msg >= WM_CTLCOLORMSGBOX && msg <= WM_CTLCOLORSTATIC) || |
| msg == WM_CTLCOLOR || msg == WM_COMPAREITEM || |
| msg == WM_VKEYTOITEM || msg == WM_CHARTOITEM || |
| msg == WM_QUERYDRAGICON || msg == WM_INITDIALOG) |
| return fResult; |
| |
| return GetWindowLongA( hwnd, DWL_MSGRESULT ); |
| } |
| |
| /*********************************************************************** |
| * DefDlgProc (USER.308) |
| */ |
| LRESULT WINAPI DefDlgProc16( HWND16 hwnd, UINT16 msg, WPARAM16 wParam, |
| LPARAM lParam ) |
| { |
| WNDPROC16 dlgproc; |
| HWND hwnd32 = WIN_Handle32( hwnd ); |
| BOOL result = FALSE; |
| |
| SetWindowLongW( hwnd32, DWL_MSGRESULT, 0 ); |
| |
| if ((dlgproc = (WNDPROC16)DEFDLG_GetDlgProc( hwnd32 ))) |
| { |
| /* Call dialog procedure */ |
| result = CallWindowProc16( dlgproc, hwnd, msg, wParam, lParam ); |
| /* 16 bit dlg procs only return BOOL16 */ |
| if( WINPROC_GetProcType( dlgproc ) == WIN_PROC_16 ) |
| result = LOWORD(result); |
| } |
| |
| if (!result && IsWindow(hwnd32)) |
| { |
| /* callback didn't process this message */ |
| |
| switch(msg) |
| { |
| case WM_ERASEBKGND: |
| case WM_SHOWWINDOW: |
| case WM_ACTIVATE: |
| case WM_SETFOCUS: |
| case DM_SETDEFID: |
| case DM_GETDEFID: |
| case WM_NEXTDLGCTL: |
| case WM_GETFONT: |
| case WM_CLOSE: |
| case WM_NCDESTROY: |
| case WM_ENTERMENULOOP: |
| case WM_LBUTTONDOWN: |
| case WM_NCLBUTTONDOWN: |
| return DEFDLG_Proc( hwnd32, msg, (WPARAM)wParam, lParam, DIALOG_get_info(hwnd32) ); |
| case WM_INITDIALOG: |
| case WM_VKEYTOITEM: |
| case WM_COMPAREITEM: |
| case WM_CHARTOITEM: |
| break; |
| |
| default: |
| return DefWindowProc16( hwnd, msg, wParam, lParam ); |
| } |
| } |
| return DEFDLG_Epilog( hwnd32, msg, result); |
| } |
| |
| |
| /*********************************************************************** |
| * DefDlgProcA (USER32.@) |
| */ |
| LRESULT WINAPI DefDlgProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) |
| { |
| WNDPROC dlgproc; |
| BOOL result = FALSE; |
| |
| SetWindowLongW( hwnd, DWL_MSGRESULT, 0 ); |
| |
| if ((dlgproc = DEFDLG_GetDlgProc( hwnd ))) |
| { |
| /* Call dialog procedure */ |
| result = CallWindowProcA( dlgproc, hwnd, msg, wParam, lParam ); |
| /* 16 bit dlg procs only return BOOL16 */ |
| if( WINPROC_GetProcType( dlgproc ) == WIN_PROC_16 ) |
| result = LOWORD(result); |
| } |
| |
| if (!result && IsWindow(hwnd)) |
| { |
| /* callback didn't process this message */ |
| |
| switch(msg) |
| { |
| case WM_ERASEBKGND: |
| case WM_SHOWWINDOW: |
| case WM_ACTIVATE: |
| case WM_SETFOCUS: |
| case DM_SETDEFID: |
| case DM_GETDEFID: |
| case WM_NEXTDLGCTL: |
| case WM_GETFONT: |
| case WM_CLOSE: |
| case WM_NCDESTROY: |
| case WM_ENTERMENULOOP: |
| case WM_LBUTTONDOWN: |
| case WM_NCLBUTTONDOWN: |
| return DEFDLG_Proc( hwnd, msg, wParam, lParam, DIALOG_get_info(hwnd) ); |
| case WM_INITDIALOG: |
| case WM_VKEYTOITEM: |
| case WM_COMPAREITEM: |
| case WM_CHARTOITEM: |
| break; |
| |
| default: |
| return DefWindowProcA( hwnd, msg, wParam, lParam ); |
| } |
| } |
| return DEFDLG_Epilog(hwnd, msg, result); |
| } |
| |
| |
| /*********************************************************************** |
| * DefDlgProcW (USER32.@) |
| */ |
| LRESULT WINAPI DefDlgProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam ) |
| { |
| BOOL result = FALSE; |
| WNDPROC dlgproc; |
| |
| SetWindowLongW( hwnd, DWL_MSGRESULT, 0 ); |
| |
| if ((dlgproc = DEFDLG_GetDlgProc( hwnd ))) |
| { |
| /* Call dialog procedure */ |
| result = CallWindowProcW( dlgproc, hwnd, msg, wParam, lParam ); |
| /* 16 bit dlg procs only return BOOL16 */ |
| if( WINPROC_GetProcType( dlgproc ) == WIN_PROC_16 ) |
| result = LOWORD(result); |
| } |
| |
| if (!result && IsWindow(hwnd)) |
| { |
| /* callback didn't process this message */ |
| |
| switch(msg) |
| { |
| case WM_ERASEBKGND: |
| case WM_SHOWWINDOW: |
| case WM_ACTIVATE: |
| case WM_SETFOCUS: |
| case DM_SETDEFID: |
| case DM_GETDEFID: |
| case WM_NEXTDLGCTL: |
| case WM_GETFONT: |
| case WM_CLOSE: |
| case WM_NCDESTROY: |
| case WM_ENTERMENULOOP: |
| case WM_LBUTTONDOWN: |
| case WM_NCLBUTTONDOWN: |
| return DEFDLG_Proc( hwnd, msg, wParam, lParam, DIALOG_get_info(hwnd) ); |
| case WM_INITDIALOG: |
| case WM_VKEYTOITEM: |
| case WM_COMPAREITEM: |
| case WM_CHARTOITEM: |
| break; |
| |
| default: |
| return DefWindowProcW( hwnd, msg, wParam, lParam ); |
| } |
| } |
| return DEFDLG_Epilog(hwnd, msg, result); |
| } |