|  | /* | 
|  | * Default dialog procedure | 
|  | * | 
|  | * Copyright 1993, 1996 Alexandre Julliard | 
|  | * | 
|  | * 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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | 
|  | */ | 
|  |  | 
|  | #include "windef.h" | 
|  | #include "winbase.h" | 
|  | #include "wingdi.h" | 
|  | #include "wine/winuser16.h" | 
|  | #include "controls.h" | 
|  | #include "win.h" | 
|  | #include "winproc.h" | 
|  | #include "wine/debug.h" | 
|  |  | 
|  | WINE_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 ) | 
|  | { | 
|  | DWORD dlgcode=0; /* initialize just to avoid a warning */ | 
|  | if (hwndNew && | 
|  | !((dlgcode=SendMessageW(hwndNew, WM_GETDLGCODE, 0, 0 )) | 
|  | & (DLGC_UNDEFPUSHBUTTON | DLGC_BUTTON))) | 
|  | 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) | 
|  | { | 
|  | if(dlgcode==DLGC_UNDEFPUSHBUTTON) | 
|  | 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); | 
|  | } |