WinForm IME输入法BUG完美修复

时间:2023-03-09 02:43:48
WinForm IME输入法BUG完美修复

本文来自http://hi.baidu.com/wingingbob/item/a2cb3fc0fe3bd1bb0d0a7b5b

《WinForm IME输入法BUG测试》里,我描述了在.NET Framework 2.0的WinForm中,中文输入法的BUG。这个BUG由来已久,听说在最新的VS2010中也没有真正得到解决。在文章后面,我怀疑是那些产生此类BUG的中文输入法本身在设计上存在的缺陷,才导致了WinForm无法正确识别。不管怎么样,问题出了就得想办法解决,今天的方法是一个“简单而有效的”解决办法,就是在窗体加载时将输入法预置为ImeMode.OnHalf。最终的结果如上图所示,王码五笔可以正确切换出来,而且以半角的方式显示(ImeMode.OnHalf不同于ImeMode.On的地方),只是默认使用英文标点符号。但是你不用担心,主流的中文输入法(如搜狗拼音)会默认采用中文标点的。代码片段如下:

  1. protected override void OnLoad(EventArgs e)
  2. {
  3. KeyPreview = true;
  4. DrawTextboxes();
  5. // 让输入法为开启半角状态
  6. ImeMode = ImeMode.OnHalf;
  7. base.OnLoad(e);
  8. }

完美修复

现在是通过对imm32.dll API调用,使之输入法状态为开启,这样就保证了WinForm程序其它窗口的输入法状态也正确显示。重写第一个窗口的OnActivited事件即可。注意,不要再使用Control的ImeMode属性了。测试下面代码,点击按钮打开一个新的窗口,仍然可以正确使用输入法。OK,问题得到完美解决~~!

  1. /* WinForm IME输入法BUG完美修复
  2. * 编译:csc.exe /target:winexe WinformImeBugFixed.cs
  3. */
  4. using System;
  5. using System.Windows.Forms;
  6. using System.Drawing;
  7. using System.Runtime.InteropServices;
  8. namespace WinformImeBugFixed
  9. {
  10. public class Form1 : Form
  11. {
  12. #region 解决输入法BUG
  13. //解决输入法BUG
  14. [DllImport("imm32.dll")]
  15. public static extern IntPtr ImmGetContext(IntPtr hwnd);
  16. [DllImport("imm32.dll")]
  17. public static extern bool ImmSetOpenStatus(IntPtr himc, bool b);
  18. protected override void OnActivated(EventArgs e)
  19. {
  20. base.OnActivated(e);
  21. IntPtr HIme = ImmGetContext(this.Handle);
  22. ImmSetOpenStatus(HIme, true);
  23. }
  24. #endregion
  25. #region 不感兴趣的
  26. private void DrawTextboxes()
  27. {
  28. Controls.Clear();
  29. int x, y, d;
  30. x = y = d = 10;
  31. for (int i = 0; i < 2; i++)
  32. {
  33. var textbox = new TextBox()
  34. {
  35. Width = 200,
  36. Location = new Point(x, y)
  37. };
  38. y += textbox.Height + d;
  39. textbox.DataBindings.Add("Text", textbox, "ImeMode");
  40. Controls.Add(textbox);
  41. }
  42. }
  43. private void DrawButton()
  44. {
  45. var button = new Button()
  46. {
  47. Text = "Show Form2",
  48. Location = new Point(10, 70)
  49. };
  50. button.Click += delegate
  51. {
  52. var form2 = new Form();
  53. form2.Text = "Form2";
  54. var textbox = new TextBox()
  55. {
  56. Width = 200,
  57. Location = new Point(10, 10)
  58. };
  59. form2.Controls.Add(textbox);
  60. form2.Show();
  61. };
  62. Controls.Add(button);
  63. }
  64. protected override void OnLoad(EventArgs e)
  65. {
  66. Text = "IME输入法BUG修复 F5-刷新 F1-博客";
  67. KeyPreview = true;
  68. DrawTextboxes();
  69. DrawButton();
  70. base.OnLoad(e);
  71. }
  72. public Form1()
  73. {
  74. InitializeComponent();
  75. }
  76. protected override void OnKeyDown(KeyEventArgs e)
  77. {
  78. try { HandleKeyDown(e); }
  79. finally { base.OnKeyDown(e); }
  80. }
  81. private void HandleKeyDown(KeyEventArgs e)
  82. {
  83. if (e.KeyCode == Keys.F5) DrawTextboxes();
  84. else if (e.KeyCode == Keys.F1) NavigateBlog();
  85. }
  86. private void NavigateBlog()
  87. {
  88. System.Diagnostics.Process.Start("http://hi.baidu.com/wingingbob/blog/item/20741734532af846251f14f1.html");
  89. }
  90. #endregion
  91. #region Form1设计器
  92. private System.ComponentModel.IContainer components = null;
  93. protected override void Dispose(bool disposing)
  94. {
  95. if (disposing && (components != null))
  96. {
  97. components.Dispose();
  98. }
  99. base.Dispose(disposing);
  100. }
  101. private void InitializeComponent()
  102. {
  103. this.SuspendLayout();
  104. this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 12F);
  105. this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
  106. this.ClientSize = new System.Drawing.Size(240, 100);
  107. this.Name = "Form1";
  108. this.Text = "Form1";
  109. this.ResumeLayout(false);
  110. }
  111. #endregion
  112. #region 入口点
  113. static class Program
  114. {
  115. [STAThread]
  116. static void Main()
  117. {
  118. Application.EnableVisualStyles();
  119. Application.SetCompatibleTextRenderingDefault(false);
  120. Application.Run(new Form1());
  121. }
  122. }
  123. #endregion
  124. }
  125. }

补充:采用OnActivited事件激活输入法,那么这个窗体的TopMost属性应该为false才能使输入法设置有效。这个原因可能是由于设置TopMost为True时,将此句代码写入构造函数里,而此时修复输入法的代码还没有被执行。也就是说,将TopMost=True的代码写在窗体的OnLoad事件中,就没有问题了。于是,我把输入法修复的代码修改如下:

  1. #region 解决输入法BUG
  2. [DllImport("imm32.dll")]
  3. public static extern IntPtr ImmGetContext(IntPtr hwnd);
  4. [DllImport("imm32.dll")]
  5. public static extern bool ImmSetOpenStatus(IntPtr himc, bool b);
  6. delegate void fixImeDele();
  7. protected override void OnLoad(EventArgs e)
  8. {
  9. fixImeDele fixime = delegate
  10. {
  11. IntPtr HIme = ImmGetContext(this.Handle);
  12. ImmSetOpenStatus(HIme, true);
  13. };
  14. this.BeginInvoke(fixime);
  15. this.TopMost = true;
  16. base.OnLoad(e);
  17. }
  18. #endregion

注意这里使用了一个小技巧,用包含BeginInvoke语句的OnLoad方法代替了原先的OnActivited方法,TopMost=True也写在这个OnLoad重载里,这样,窗口置顶和输入法开启的代码全部有效。BTW,在Onload中加入BeginInvoke的方式还可以解决Control.Focus()方法在Load中失效的问题,原理和这个一样,写在委托里调用就是了,范例参见:《WinForm IME输入法BUG测试》