最近发现,之前解决的输入法问题在Windows 8.1下又不工作了:
- 微软拼音输入法表现为:首字母确实,字符残留。
- 百度拼音输入法表现为:字符乱序
- QQ拼音输入法表现为:字符乱序
- Sogou拼音输入法表现为:输入字符消失
各个输入表现不一,吐血!研究了好几天,尝试着在程序中加入对TSF的支持,都告失败。无奈,只能联系了微软技术支持,在一周之后,他们给出了解决方案,很简单的方案:
在Cancel Compostion和切换焦点之间,停顿一下(说白了,就是Sleep),呵呵,这个不看他们的源码,肯定是想不出来的。那么,具体停顿多少时间呢???测试结果如下:微软拼音输入法表现为:首字母确实,字符残留。
- 微软拼音输入法:需要1秒钟(1000毫秒)
- 其他输入法:300毫秒
这微软输入法到底搞的是哪出。。。1秒钟的停顿很难接受,另外,首字母缺失的问题,也没能解决,所以呢。。。继续劳烦微软想办法咯。。。
先贴一下基于微软目前解决方案的代码,将来有更好的方案再更新~
private void listView1_KeyDown(object sender, KeyEventArgs e) { // Avoid ALT+` to change the focus if (!e.Alt) { CancelComposition(); Thread.Sleep(1000); CommandLine.GiveFocus(); //if (mEatenVirtualKey != 229) //{ // SendKeys.SendWait(Convert.ToChar(mEatenVirtualKey).ToString().ToLower()); //} } } [DllImport("Imm32.dll")] internal static extern IntPtr ImmGetContext(IntPtr hWnd); [DllImport("Imm32.dll")] [return: MarshalAs(UnmanagedType.Bool)] internal static extern bool ImmReleaseContext(IntPtr hWnd, IntPtr hIMC); [DllImport("Imm32.dll", SetLastError = true)] [return: MarshalAs(UnmanagedType.Bool)] internal static extern bool ImmNotifyIME(IntPtr hIMC, uint action, uint index, int value); private void CancelComposition() { IntPtr hIMC = ImmGetContext(this.Handle); try { ImmNotifyIME(hIMC, 0x0015, 4, 0); } finally { ImmReleaseContext(this.Handle, hIMC); } }
继续做优化:
- 在没有启用输入法时,不应该有停顿。
- 输入法切换到英文输入模式时,不应该有停顿。
bool IsImeOn() { HRESULT hr = S_OK; CComPtr<ITfInputProcessorProfiles> pProfiles; LANGID langid; BSTR bstrImeName = NULL; hr = CoCreateInstance(CLSID_TF_InputProcessorProfiles, NULL, CLSCTX_INPROC_SERVER, IID_ITfInputProcessorProfiles, (LPVOID*)&pProfiles); hr = pProfiles->GetCurrentLanguage(&langid); CLSID textSrvId, profileId; hr = pProfiles->GetDefaultLanguageProfile(langid, GUID_TFCAT_TIP_KEYBOARD, &textSrvId, &profileId); LANGID activeLangId; CLSID activeProfileId; hr = pProfiles->GetActiveLanguageProfile(textSrvId, &activeLangId, &activeProfileId); return activeLangId == 0 } bool IsImeInEnglishMode() { HRESULT hr = S_OK; ITfThreadMgr* pThreadMgr; hr = CoCreateInstance( CLSID_TF_ThreadMgr, NULL, CLSCTX_INPROC_SERVER, IID_ITfThreadMgr, (void**)&pThreadMgr); ASSERT(S_OK == hr); ITfCompartmentMgr* pCompMgr; hr = pThreadMgr->QueryInterface(IID_ITfCompartmentMgr, (void**)&pCompMgr); ASSERT(S_OK == hr); ITfCompartment* pcomp; hr = pCompMgr->GetCompartment(GUID_COMPARTMENT_KEYBOARD_INPUTMODE_CONVERSION, &pcomp); ASSERT(S_OK == hr); VARIANT var; VariantInit(&var); hr = pcomp->GetValue(&var); ASSERT(S_OK == hr); return var.lVal == 0x904; }
代码可能有点问题,将来再更新