解决 MFC ActiveX 控件输入法窗口不跟随问题
一. 背景
我前段时间写了一个输入控件, 发现在输入中文的时候, 输入法候选窗口不跟随输入光标, 如下图所示
输入法候选窗口在显示器最左上角, 而输入控件在其他地方. 这个现象对于强迫症来说超级不爽, 所以必须纠正
二. 解决方法
1. 添加 DefWindowProc 虚函数
通过类向导添加 DefWindowProc 虚函数
2. 在 DefWindowProc 中添加代码
DefWindowProc 添加如下代码就 OK 了
// TODO: 在此添加专用代码和/或调用基类
HIMC hImc = nullptr;
switch (message)
{
case WM_IME_COMPOSITION:
hImc = ImmGetContext(m_hWnd);
if (nullptr != hImc)
{
COMPOSITIONFORM cf;
cf.dwStyle = CFS_CANDIDATEPOS;
// 这里的 cf.ptCurrentPos 是输入光标 Caret 的位置, 就是用 SetCaretPos 设置的位置
// 在我的代码中有用 m_EditInfo.ptCaret 来记录, 所以这个变量是你没有的
cf.ptCurrentPos.x = m_EditInfo.ptCaret.x;
cf.ptCurrentPos.y = m_EditInfo.ptCaret.y + 32; // 32 你可以改成其他的数字或者动态计算
if (ImmSetCompositionWindow(hImc, &cf))
{
ImmReleaseContext(m_hWnd, hImc);
}
}
break;
default:
break;
}
return COleControl::DefWindowProc(message, wParam, lParam);
完成之后, 输入法就跟随输入光标的位置了
三. 禁用输入法
有的时候只想输入英文字母或者数字, 或者输入密码的时候不想用其他输入法, 这个功能就很方便使用者了, 不用来回切换输入法. 比如上面的测试控件我设置成密码模式, 但是如果在中文输入状态下, 就会显示烦人的输入法候选框
要解决这个问题也很简单, 只需要三步
1. 添加 HIMC 类型的成员变量
添加 HIMC 类型的成员变量 m_hImc, 并在构造函数中初始化成 nullptr
2. 添加 WM_SETFOCUS 消息
通过类向导添加 WM_SETFOCUS 消息, 在消息响应函数中添加如下代码
// TODO: 在此处添加消息处理程序代码
// if 中的条件判断是我代码中的宏定义, 你也没有, 你要用其他的方式判断或者不用判断
// m_hImc 是 HIMC 类型的成员变量
if (ET_PASSWORD == m_nType || IS_NUMBER_ONLY)
{
m_hImc = ImmAssociateContext(m_hWnd, nullptr);
}
3. 添加 WM_KILLFOCUS 消息
通过类向导添加 WM_KILLFOCUS 消息, 在消息响应函数中添加如下代码
// TODO: 在此处添加消息处理程序代码
if (nullptr != m_hImc)
{
ImmAssociateContext(m_hWnd, m_hImc);
m_hImc = nullptr;
}
经过上面三步后, 输入法就禁用了, 效果如下