|  | /* | 
|  | * Default dialog procedure | 
|  | * | 
|  | * Copyright 1993, 1996 Alexandre Julliard | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include "windef.h" | 
|  | #include "winbase.h" | 
|  | #include "wingdi.h" | 
|  | #include "wine/winuser16.h" | 
|  | #include "dialog.h" | 
|  | #include "win.h" | 
|  | #include "winproc.h" | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           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 (SendMessage16( hwndPrev, WM_GETDLGCODE, 0, 0 ) & DLGC_HASSETSEL) | 
|  | SendMessage16( hwndPrev, EM_SETSEL16, TRUE, MAKELONG( -1, 0 ) ); | 
|  | } | 
|  | if (SendMessage16( hwndCtrl, WM_GETDLGCODE, 0, 0 ) & DLGC_HASSETSEL) | 
|  | SendMessage16( hwndCtrl, EM_SETSEL16, FALSE, MAKELONG( 0, -1 ) ); | 
|  | SetFocus( hwndCtrl ); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           DEFDLG_SaveFocus | 
|  | */ | 
|  | static BOOL DEFDLG_SaveFocus( HWND hwnd, DIALOGINFO *infoPtr ) | 
|  | { | 
|  | HWND hwndFocus = GetFocus(); | 
|  |  | 
|  | if (!hwndFocus || !IsChild( hwnd, hwndFocus )) return FALSE; | 
|  | infoPtr->hwndFocus = hwndFocus; | 
|  | /* Remove default button */ | 
|  | return TRUE; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           DEFDLG_RestoreFocus | 
|  | */ | 
|  | static BOOL DEFDLG_RestoreFocus( HWND hwnd, DIALOGINFO *infoPtr ) | 
|  | { | 
|  | if (!infoPtr->hwndFocus || IsIconic(hwnd)) return FALSE; | 
|  | if (!IsWindow( infoPtr->hwndFocus )) return FALSE; | 
|  |  | 
|  | /* Don't set the focus back to controls if EndDialog is already called.*/ | 
|  | if (!(infoPtr->flags & DF_END)) | 
|  | DEFDLG_SetFocus( hwnd, infoPtr->hwndFocus ); | 
|  |  | 
|  | /* This used to set infoPtr->hwndFocus to NULL for no apparent reason, | 
|  | sometimes losing focus when receiving WM_SETFOCUS messages. */ | 
|  | return TRUE; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           DEFDLG_FindDefButton | 
|  | * | 
|  | * Find the current default push-button. | 
|  | */ | 
|  | static HWND DEFDLG_FindDefButton( HWND hwndDlg ) | 
|  | { | 
|  | HWND hwndChild = GetWindow( hwndDlg, GW_CHILD ); | 
|  | while (hwndChild) | 
|  | { | 
|  | if (SendMessage16( 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 && | 
|  | !(SendMessage16(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: | 
|  | FillWindow16( hwnd, hwnd, (HDC16)wParam, (HBRUSH16)CTLCOLOR_DLG ); | 
|  | return 1; | 
|  |  | 
|  | case WM_NCDESTROY: | 
|  |  | 
|  | /* Free dialog heap (if created) */ | 
|  | if (dlgInfo->hDialogHeap) | 
|  | { | 
|  | GlobalUnlock16(dlgInfo->hDialogHeap); | 
|  | GlobalFree16(dlgInfo->hDialogHeap); | 
|  | dlgInfo->hDialogHeap = 0; | 
|  | } | 
|  |  | 
|  | /* Delete font */ | 
|  | if (dlgInfo->hUserFont) | 
|  | { | 
|  | DeleteObject( dlgInfo->hUserFont ); | 
|  | dlgInfo->hUserFont = 0; | 
|  | } | 
|  |  | 
|  | /* Delete menu */ | 
|  | if (dlgInfo->hMenu) | 
|  | { | 
|  | DestroyMenu( dlgInfo->hMenu ); | 
|  | dlgInfo->hMenu = 0; | 
|  | } | 
|  |  | 
|  | /* Delete window procedure */ | 
|  | WINPROC_FreeProc( dlgInfo->dlgProc, WIN_PROC_WINDOW ); | 
|  | dlgInfo->dlgProc = (HWINDOWPROC)0; | 
|  | dlgInfo->flags |= DF_END;  /* just in case */ | 
|  |  | 
|  | /* Window clean-up */ | 
|  | return DefWindowProcA( hwnd, msg, wParam, lParam ); | 
|  |  | 
|  | case WM_SHOWWINDOW: | 
|  | if (!wParam) DEFDLG_SaveFocus( hwnd, dlgInfo ); | 
|  | return DefWindowProcA( hwnd, msg, wParam, lParam ); | 
|  |  | 
|  | case WM_ACTIVATE: | 
|  | if (wParam) DEFDLG_RestoreFocus( hwnd, dlgInfo ); | 
|  | else DEFDLG_SaveFocus( hwnd, dlgInfo ); | 
|  | return 0; | 
|  |  | 
|  | case WM_SETFOCUS: | 
|  | DEFDLG_RestoreFocus( hwnd, dlgInfo ); | 
|  | 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) | 
|  | { | 
|  | WND *wnd = WIN_FindWndPtr( hwndFocus ); | 
|  |  | 
|  | if( wnd ) | 
|  | { | 
|  | /* always make combo box hide its listbox control */ | 
|  |  | 
|  | if( WIDGETS_IsControl( wnd, BIC32_COMBO ) ) | 
|  | SendMessageA( hwndFocus, CB_SHOWDROPDOWN, | 
|  | FALSE, 0 ); | 
|  | else if( WIDGETS_IsControl( wnd, BIC32_EDIT ) && | 
|  | WIDGETS_IsControl( wnd->parent, | 
|  | BIC32_COMBO )) | 
|  | SendMessageA( wnd->parent->hwndSelf, | 
|  | CB_SHOWDROPDOWN, FALSE, 0 ); | 
|  | } | 
|  | WIN_ReleaseWndPtr(wnd); | 
|  | } | 
|  | } | 
|  | 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(DIALOGINFO* dlgInfo, 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 dlgInfo->msgResult; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           DefDlgProc16   (USER.308) | 
|  | */ | 
|  | LRESULT WINAPI DefDlgProc16( HWND16 hwnd, UINT16 msg, WPARAM16 wParam, | 
|  | LPARAM lParam ) | 
|  | { | 
|  | DIALOGINFO * dlgInfo; | 
|  | BOOL result = FALSE; | 
|  | WND * wndPtr = WIN_FindWndPtr( hwnd ); | 
|  |  | 
|  | if (!wndPtr) return 0; | 
|  | dlgInfo = (DIALOGINFO *)&wndPtr->wExtra; | 
|  | dlgInfo->msgResult = 0; | 
|  |  | 
|  | if (dlgInfo->dlgProc) {	/* Call dialog procedure */ | 
|  | result = CallWindowProc16( (WNDPROC16)dlgInfo->dlgProc, | 
|  | hwnd, msg, wParam, lParam ); | 
|  | /* 16 bit dlg procs only return BOOL16 */ | 
|  | if( WINPROC_GetProcType( dlgInfo->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: | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DEFDLG_Proc( (HWND)hwnd, msg, | 
|  | (WPARAM)wParam, lParam, dlgInfo ); | 
|  | case WM_INITDIALOG: | 
|  | case WM_VKEYTOITEM: | 
|  | case WM_COMPAREITEM: | 
|  | case WM_CHARTOITEM: | 
|  | break; | 
|  |  | 
|  | default: | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DefWindowProc16( hwnd, msg, wParam, lParam ); | 
|  | } | 
|  | } | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DEFDLG_Epilog(dlgInfo, msg, result); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           DefDlgProcA   (USER32.120) | 
|  | */ | 
|  | LRESULT WINAPI DefDlgProcA( HWND hwnd, UINT msg, | 
|  | WPARAM wParam, LPARAM lParam ) | 
|  | { | 
|  | DIALOGINFO * dlgInfo; | 
|  | BOOL result = FALSE; | 
|  | WND * wndPtr = WIN_FindWndPtr( hwnd ); | 
|  |  | 
|  | if (!wndPtr) return 0; | 
|  | dlgInfo = (DIALOGINFO *)&wndPtr->wExtra; | 
|  | dlgInfo->msgResult = 0; | 
|  |  | 
|  | if (dlgInfo->dlgProc) {      /* Call dialog procedure */ | 
|  | result = CallWindowProcA( (WNDPROC)dlgInfo->dlgProc, | 
|  | hwnd, msg, wParam, lParam ); | 
|  | /* 16 bit dlg procs only return BOOL16 */ | 
|  | if( WINPROC_GetProcType( dlgInfo->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: | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DEFDLG_Proc( (HWND)hwnd, msg, | 
|  | (WPARAM)wParam, lParam, dlgInfo ); | 
|  | case WM_INITDIALOG: | 
|  | case WM_VKEYTOITEM: | 
|  | case WM_COMPAREITEM: | 
|  | case WM_CHARTOITEM: | 
|  | break; | 
|  |  | 
|  | default: | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DefWindowProcA( hwnd, msg, wParam, lParam ); | 
|  | } | 
|  | } | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DEFDLG_Epilog(dlgInfo, msg, result); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           DefDlgProcW   (USER32.121) | 
|  | */ | 
|  | LRESULT WINAPI DefDlgProcW( HWND hwnd, UINT msg, WPARAM wParam, | 
|  | LPARAM lParam ) | 
|  | { | 
|  | DIALOGINFO * dlgInfo; | 
|  | BOOL result = FALSE; | 
|  | WND * wndPtr = WIN_FindWndPtr( hwnd ); | 
|  |  | 
|  | if (!wndPtr) return 0; | 
|  | dlgInfo = (DIALOGINFO *)&wndPtr->wExtra; | 
|  | dlgInfo->msgResult = 0; | 
|  |  | 
|  | if (dlgInfo->dlgProc) {      /* Call dialog procedure */ | 
|  | result = CallWindowProcW( (WNDPROC)dlgInfo->dlgProc, | 
|  | hwnd, msg, wParam, lParam ); | 
|  | /* 16 bit dlg procs only return BOOL16 */ | 
|  | if( WINPROC_GetProcType( dlgInfo->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: | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DEFDLG_Proc( (HWND)hwnd, msg, | 
|  | (WPARAM)wParam, lParam, dlgInfo ); | 
|  | case WM_INITDIALOG: | 
|  | case WM_VKEYTOITEM: | 
|  | case WM_COMPAREITEM: | 
|  | case WM_CHARTOITEM: | 
|  | break; | 
|  |  | 
|  | default: | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DefWindowProcW( hwnd, msg, wParam, lParam ); | 
|  | } | 
|  | } | 
|  | WIN_ReleaseWndPtr(wndPtr); | 
|  | return DEFDLG_Epilog(dlgInfo, msg, result); | 
|  | } |