Click here to Skip to main content
15,860,972 members
Articles / Desktop Programming / MFC
Article

Keyboard messages/accelerators handling in MFC dialog based applications

Rate me:
Please Sign up or sign in to vote.
4.76/5 (81 votes)
1 Dec 2001CPOL3 min read 501.1K   102   98
This article explains how you can override PreTranslateMessage and ProcessMessageFilter in dialog based apps

Introduction

There are a substantial number of Windows programmers who insist, often very vehemently, that a programmer should avoid overriding PreTranslateMessage. They have their reasons for saying so and I believe they are correct. But in this article my intention is not to contemplate on whether PreTranslateMessage is good for you or whether you should avoid it like the plague. I have found that PreTranslateMessage can come in quite handy in dialog-based applications for handling keyboard messages. In addition to using PreTranslateMessage I also show you how you can override ProcessMessageFilter for handling accelerator keys in a dialog based application.

Using PreTranslateMessage to handle dialog keystrokes

Very often you hear questions from novice programmers asking how they can trap keystrokes in a dialog based application. Presumably they tried to handle WM_KEYDOWN/WM_KEYUP unsuccessfully. The whole problem is that in a dialog based application the focus is always on one of the child controls and not on the main dialog window. So what do you need to do? You need to override PreTranslateMessage. I'll show you a simple example.

Suppose that you have a dialog based app with a lot of edit boxes on the dialog. It's basically a data entry program and thus you feel it would make it easier for the end-user if pressing the ENTER key would take the focus to the next edit box, just as if he had pressed TAB. The solution is so very easy and straightforward with PreTranslateMessage as I'll demonstrate below.

BOOL CPreTransTestDlg::PreTranslateMessage(MSG* pMsg) 
{
    if(pMsg->message==WM_KEYDOWN)
    {
        if(pMsg->wParam==VK_RETURN)
            pMsg->wParam=VK_TAB;
    }	
    return CDialog::PreTranslateMessage(pMsg);
}

All I have done is to check whether the message is a WM_KEYDOWN, and if it is so, then I check to see if the wParam is VK_RETURN. If I find it so, I change the wParam to VK_TAB and then the base class implementation is called. Easy huh?

Using ProcessMessageFilter to handle dialog-based accelerator keys

Let's say you have a menu in your dialog based app. And you have an accelerator key for some particular task. You'll soon be disappointed to find that the hotkey does not work. The problem is that the modal dialog app's message loop does not call TranslateAccelerator. I do not know why this is so. Presumable the Microsoft team decided that people shouldn't use dialog based apps to write complicated applications, with hotkeys and menus.

But as usual they have suggested a workaround too. Here's is how you go about implementing it. I'd like to state again, that even though this is a Microsoft recommended technique there will be a good majority of MFC gurus, like Joseph Newcomer for example, who would tell you that you shouldn't be doing this. But then sometimes you have to sacrifice elegance for getting things done quickly and with minimum effort.

  • Add a member variable to your CWinApp derived class.
  • HACCEL m_haccel;
  • Use the resource editor to create a new Accelerator, by default it will be named IDR_ACCELERATOR1. And add a new accelerator key that is a short cut for some menu item.
  • Put the following line in your InitInstance just before the line where the CDialog derived object is declared
  • m_haccel=LoadAccelerators(AfxGetInstanceHandle(), 
            MAKEINTRESOURCE(IDR_ACCELERATOR1));
  • Now override ProcessMessageFilter and modify the function so that it looks like :-
    BOOL CPreTransTestApp::ProcessMessageFilter(int code, LPMSG lpMsg) 
    {
        if(m_haccel)
        {
            if (::TranslateAccelerator(m_pMainWnd->m_hWnd, m_haccel, lpMsg)) 
                return(TRUE);
        }
    	
        return CWinApp::ProcessMessageFilter(code, lpMsg);
    }

All we did was to call TranslateAccelerator and if it succeeds then we don't need to call the base class ProcessMessageFilter, as the message has been handled. So we return TRUE.

Disclaimer

The author wishes to state here that the two methods mentioned above are generally used methods and the author is not in any way endorsing these methods. Users should read more on the usage of PreTranslateMessage and ProcessMessageFilter before they use it in their programs.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
United States United States
Nish Nishant is a technology enthusiast from Columbus, Ohio. He has over 20 years of software industry experience in various roles including Chief Technology Officer, Senior Solution Architect, Lead Software Architect, Principal Software Engineer, and Engineering/Architecture Team Leader. Nish is a 14-time recipient of the Microsoft Visual C++ MVP Award.

Nish authored C++/CLI in Action for Manning Publications in 2005, and co-authored Extending MFC Applications with the .NET Framework for Addison Wesley in 2003. In addition, he has over 140 published technology articles on CodeProject.com and another 250+ blog articles on his WordPress blog. Nish is experienced in technology leadership, solution architecture, software architecture, cloud development (AWS and Azure), REST services, software engineering best practices, CI/CD, mentoring, and directing all stages of software development.

Nish's Technology Blog : voidnish.wordpress.com

Comments and Discussions

 
QuestionRestrict LeftWindowKey(VK_LWIN+D) operation Pin
chetan7866-Apr-15 2:23
chetan7866-Apr-15 2:23 
GeneralMy vote of 3 Pin
SharanyaMahi1-Feb-12 0:54
SharanyaMahi1-Feb-12 0:54 
Thanks a lot for the information..
GeneralMy vote of 5 Pin
Rethmeier27-Apr-11 22:26
Rethmeier27-Apr-11 22:26 
GeneralMy vote of 5 Pin
Member 774968413-Apr-11 11:36
Member 774968413-Apr-11 11:36 
GeneralMy vote of 5 Pin
Gokulnath0073-Jan-11 21:09
Gokulnath0073-Jan-11 21:09 
GeneralAmazing Pin
vikas02312-Nov-09 0:45
vikas02312-Nov-09 0:45 
Generalthanks! Pin
anowsober19-Jan-09 17:30
anowsober19-Jan-09 17:30 
GeneralAccelerator for child dialog Pin
hari_honey20-Aug-08 3:01
hari_honey20-Aug-08 3:01 
GeneralRe: Accelerator for child dialog Pin
David Johns21-Sep-10 16:07
David Johns21-Sep-10 16:07 
GeneralCapture consecutive VK_ENTER messages Pin
mrcdsix3-Apr-08 8:13
mrcdsix3-Apr-08 8:13 
GeneralProblem with PreTranslateMessage Pin
asi1103-Oct-07 21:28
asi1103-Oct-07 21:28 
Questioni want to call click event of button with Ctrl + Up Pin
maulesh bhojani22-Aug-07 5:47
maulesh bhojani22-Aug-07 5:47 
QuestionI need Accelrator table in My Dll? Pin
ramesh_5304-May-07 2:14
ramesh_5304-May-07 2:14 
Questiononly work after minimized it....help me please. Pin
fikree15-Feb-07 14:55
fikree15-Feb-07 14:55 
QuestionHow to simulate the mouse behavior holding down a button with a key Pin
chen.zd24-Nov-06 19:56
chen.zd24-Nov-06 19:56 
Question! Accelerators Active In Child Dialogs Pin
Synetech4-Aug-06 20:42
Synetech4-Aug-06 20:42 
AnswerRe: ! Accelerators Active In Child Dialogs Pin
Synetech7-Aug-06 13:46
Synetech7-Aug-06 13:46 
General! Works Great Pin
Synetech22-Jul-06 8:20
Synetech22-Jul-06 8:20 
QuestionHow to set project for always support hots key ? Pin
Max++10-Jun-06 23:41
Max++10-Jun-06 23:41 
AnswerRe: How to set project for always support hots key ? Pin
Synetech22-Jul-06 6:34
Synetech22-Jul-06 6:34 
Generalerror C2039: 'ProcessMessageFilter' : is not a member of 'CWinApp' Pin
atrung5-Jun-06 17:53
atrung5-Jun-06 17:53 
General1000 thx! Pin
chris mook11-Apr-06 3:02
chris mook11-Apr-06 3:02 
QuestionHow to do this to a dialog under a child window? Pin
kezhu31-Mar-06 11:34
kezhu31-Mar-06 11:34 
QuestionHow to call a current Function now? Pin
bosfan11-Jan-06 2:00
bosfan11-Jan-06 2:00 
AnswerRe: How to call a current Function now? Pin
bosfan11-Jan-06 3:57
bosfan11-Jan-06 3:57 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.