9

How to select all text in edit control by pressing Ctrl+A? I can catch Ctrl+A for parent window in WndProc. But I don't know how to catch ctrl+a which are applied for edit control. Also I tried to use accelerators, but again it applies only for parent window. Thanks. EDIT: 1-st the simplest method This method Based on @phord's answers in this question: win32 select all on edit ctrl (textbox)

while( (bRet = GetMessage( &msg, NULL, 0, 0 )) != 0)
{ 
    if (bRet == -1)
    {
        // handle the error and possibly exit
    }
    else
    {
        if (msg.message == WM_KEYDOWN && msg.wParam == 'A' && GetKeyState(VK_CONTROL) < 0)
    {
        HWND hFocused = GetFocus();
        wchar_t className[6];
        GetClassName(hFocused, className, 6);
        if (hFocused && !wcsicmp(className, L"edit"))
            SendMessage(hFocused, EM_SETSEL, 0, -1);
    }
    TranslateMessage(&msg);
    DispatchMessage(&msg);
    }
}

EDIT: 2-nd method Need to use CreateAcceleratorTable + TranslateAccelerator functions:

//global variables:

enum {ID_CTRL_A = 1};
HACCEL accel;

//main procedure

ACCEL ctrl_a;
ctrl_a.cmd = ID_CTRL_A; // Hotkey ID
ctrl_a.fVirt = FCONTROL | FVIRTKEY;
ctrl_a.key = 0x41; //'A' key
accel = CreateAcceleratorTable(&ctrl_a, 1); //we have only one hotkey

//How GetMessage loop looks

while( (bRet = GetMessage( &msg, NULL, 0, 0 )) != 0)
{ 
    if (bRet == -1)
    {
        // handle the error and possibly exit
    }
    else
    {
        if (!TranslateAccelerator(hWnd, accel,  &msg))
        {
            TranslateMessage(&msg); 
            DispatchMessage(&msg); 
        }
    }
}

//in WndProc we must add next cases

case WM_COMMAND:
{
    if (LOWORD(wParam) == ID_CTRL_A && HIWORD(wParam) == 1)
    {
        //on which control there was pressed Ctrl+A
        //there is no way of getting HWND through wParam and lParam
        //so we get HWND which currently has focus.
        HWND hFocused = GetFocus();
        wchar_t className[6];
        GetClassName(hFocused, className, 6);
        if (hFocudsed && !wcsicmp(className, L"edit"))
            SendMessage(hFocused, EM_SETSEL, 0, -1);
    }
}
break;
case WM_DESTROY:
{
   DestroyAcceleratorTable(accel);
   PostQuitMessage(0);
}
break;

As you can see this is pretty simple.

4

4 回答 4

9

无需处理 WM_KEYDOWN!我知道这里的大多数示例(以及 CodeProject 和许多其他地方)都说有,但它并不能解决每当出现未处理的 WM_CHAR 时产生的蜂鸣声。

相反,试试这个:

LRESULT CALLBACK Edit_Prc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam){
  if(msg==WM_CHAR&&wParam==1){SendMessage(hwnd,EM_SETSEL,0,-1); return 1;}
  else return CallWindowProc((void*)WPA,hwnd,msg,wParam,lParam);
}

请记住使用 WPA=SetWindowLong(...) 将 EDIT 控件子类化到此 Edit_Prc() ,其中 WPA 是 CallWindowProc(...) 的窗口过程地址

于 2014-08-18T03:16:40.253 回答
6

首先更改编辑控件的 WindowProc:

if (!(pEditProc = (WNDPROC)SetWindowLong(hEdit, GWL_WNDPROC, (LONG)&EditProc)))
{
    assert(false);
    return false;
}

然后在新窗口proc中,处理ctrl+a:

LRESULT CALLBACK EditProc( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam )
{
    if (msg == WM_KEYDOWN) {
        if (GetKeyState(VK_CONTROL) & 0x8000 && wParam == 'A') {
            SendMessage(hwnd, EM_SETSEL, 0, -1);
        }
    }
    return CallWindowProc(pEditProc, hwnd, msg, wParam, lParam);
}
于 2013-01-17T03:19:07.610 回答
1

好消息!

似乎编辑控件(不是多行)现在在 Win10 上原生支持 Ctrl + A。

我当前的 Windows SDK 版本是 10.0.17763.0。

仅在使用纯 Windows API 创建的简单 GUI 应用程序上进行测试。

MFC APP 应该有相同的结果。

测试二进制平台为x86,操作系统为Win10 x64。

于 2019-03-25T06:43:54.370 回答
0

菜鸟证明版本?

我也使用加速器表编写了自己的版本。这稍微清除了 WinMain,我试图让所有东西都尽可能地证明 n00b(因为我是其中之一)。由于不需要枚举,因此也省略了枚举。

如前所述,我只是使用winapi的初学者,所以如果我错了,请务必纠正我。

在“Resource.h”中,我定义了两个 ID,一个用于我们将使用的加速器表,一个用于我们将使用的 selectall 命令。

Resource.h 内部:

#define IDR_ACCEL1 101
#define ID_SELECT_ALL 9003

然后在资源文件中(在 vs2017 中这是PROJECTNAME .rc)我们定义了加速器表。 项目名称.rc

IDR_ACCEL1 ACCELERATORS
{
    0x41, ID_SELECT_ALL, VIRTKEY, CONTROL // ctrl-A
}

描述

0x41 是 virtkey 'a'。

ID_SELECT_ALL(将是命令的 ID,这应该是我们在 Resource.h 文件中定义的 ID。

VIRTKEY 关键字指示应将 0x41 解释为虚拟键。

CONTROL 是将 a 与 (ctrl+a) 组合所需的修饰符。

然后在 WinMain 函数中加载加速器:

HACCEL hAccel = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDR_ACCEL1));
if (hAccel == NULL)
{
    MessageBox(NULL, _T("Failed to load accelerator table!"),_T("Error"), MB_OK | MB_ICONEXCLAMATION);
    return 0;
}

注意:在尝试定义 hAccel 之后,我们会检查是否分配了有效句柄。虽然这不是必需的,但我相信这是更好的约定。

之后我们将 TranslateAccelerator 函数添加到消息循环中,因此可以在窗口过程中处理命令:

BOOL bRet;
while (bRet = GetMessage(&Msg, NULL, 0, 0) > 0)
{
    if (bRet == -1)
    {
        // Error handling can be done here.
    }
    else if (!TranslateAccelerator(hwnd, hAccel, &Msg))
    {
        TranslateMessage(&Msg);
        DispatchMessage(&Msg);
    }
}

然后最后在Window过程里面 我们添加如下代码:

switch(msg)
{
    case WM_COMMAND:
       {
            switch (LOWORD(wParam))
            {
                case ID_SELECT_ALL:
                {
                    HWND hEdit;
                    hEdit = GetDlgItem(hwnd, IDC_MAIN_EDIT)  // Define IDC_MAIN_EDIT with an integer value in "Resource.h".
                }
                break;
            }
       break;
       }
}

注意:传递给 WM_COMMAND 的消息是我们为 ctrl+a 加速定义的 ID。

我希望这将有助于其他 n00bies。

于 2018-02-07T13:35:52.230 回答