VS2005中文輸入法自動轉(zhuǎn)換為全角的兩種解決方法

2010-08-28 10:49:44來源:西部e網(wǎng)作者:

      最近在用VS2005做項目的時候,一直忍受著VS2005輸入法自動切換到全角的Bug的作怪,一邊等待著微軟給我們一個解決的方案。但是,我的項目都要作為產(chǎn)品打包出去了,微軟還是悶頭不對這個Bug出一個解決方法。怎么辦?我可以忍受這個輸入法來回切換之苦,可用戶體驗可不會饒過我們的。弄不好,來個集體罷用,讓我們都到微軟喝西北風(fēng)去啊!
      總不能就這么交出產(chǎn)品出去吧,只有自己動手了。下面我用兩種方法來實現(xiàn)如何避免輸入法的這個Bug。
方法一:
Form的Pain和遍歷Control的Enter方法。
      首先,我們?yōu)榱耸鼓械拇a更簡潔,我們把所要做的步驟封裝到一個單獨的類中,類代碼如下:

 1using System;
 2using System.Runtime.InteropServices;
 3
 4namespace MyDemo
 5
 6    public static class clsIme
 7    {
 8        //聲明一些API函數(shù)
 9        [DllImport("imm32.dll")]
10        public static extern IntPtr ImmGetContext(IntPtr hwnd);
11        [DllImport("imm32.dll")]
12        public static extern bool ImmGetOpenStatus(IntPtr himc);
13        [DllImport("imm32.dll")]
14        public static extern bool ImmSetOpenStatus(IntPtr himc, bool b);
15        [DllImport("imm32.dll")]
16        public static extern bool ImmGetConversionStatus(IntPtr himc, ref int lpdw, ref int lpdw2);
17        [DllImport("imm32.dll")]
18        public static extern int ImmSimulateHotKey(IntPtr hwnd, int lngHotkey);
19        public const int IME_CMODE_FULLSHAPE = 0x8;
20        public const int IME_CHOTKEY_SHAPE_TOGGLE = 0x11;
21        //重載SetIme,傳入Form
22        public static void SetIme(Form frm)
23        {
24            frm.Paint += new PaintEventHandler(frm_Paint);
25            ChangeAllControl(frm);
26        }

27        //重載SetIme,傳入Control
28        public static void SetIme(Control ctl)
29        {
30            ChangeAllControl(ctl);
31        }

32        //重載SetIme,傳入對象句柄
33        public static void SetIme(IntPtr Handel)
34        {
35            ChangeControlIme(Handel);
36        }

37        private static void ChangeAllControl(Control ctl)
38        {
39            //在控件的的Enter事件中觸發(fā)來調(diào)整輸入法狀態(tài)
40            ctl.Enter += new EventHandler(ctl_Enter);
41            //遍歷子控件,使每個控件都用上Enter的委托處理
42            foreach (Control ctlChild in ctl.Controls)
43                ChangeAllControl(ctlChild);
44        }

45
46        static void frm_Paint(object sender, PaintEventArgs e)
47        {
48          /*有人問為什么使用Pain事件,而不用Load事件或Activated事件,是基于下列考慮:
49             * 1、在您的Form中,有些控件可能是運行時動態(tài)添加的
50             * 2、在您的Form中,使用到了非.NET的OCX控件
51             * 3、Form調(diào)用子Form的時候,Activated事件根本不會觸發(fā) */

52            ChangeControlIme(sender);
53        }

54        //控件的Enter處理程序
55        static void ctl_Enter(object sender, EventArgs e)
56        {
57            ChangeControlIme(sender);
58        }

59        private static void ChangeControlIme(object sender)
60        {
61            Control ctl = (Control)sender;
62            ChangeControlIme(ctl.Handle);
63        }

64        //下面這個函數(shù)才是真正檢查輸入法的全角半角狀態(tài)
65        private static void ChangeControlIme(IntPtr h)
66        {
67            IntPtr HIme = ImmGetContext(h);            
68            if (ImmGetOpenStatus(HIme))  //如果輸入法處于打開狀態(tài)
69            {
70                int iMode = 0;
71                int iSentence = 0;
72                bool bSuccess = ImmGetConversionStatus(HIme, ref iMode, ref iSentence);  //檢索輸入法信息
73                if (bSuccess)
74                {
75                    if ((iMode & IME_CMODE_FULLSHAPE) > 0)   //如果是全角
76                        ImmSimulateHotKey(h, IME_CHOTKEY_SHAPE_TOGGLE);  //轉(zhuǎn)換成半角
77                }

78            }

79        }

80    }

81}


      有人問為什么使用Pain事件,而不用Load事件或Activated事件,我是基于下列考慮:

      1、在您的Form中,有些控件可能是運行時動態(tài)添加的
      2、在您的Form中,使用到了非.NET的OCX控件
      3、Form調(diào)用子Form的時候,Activated事件根本不會觸發(fā)

使用這個類的方法為:
      在您的界面中,在Load的時候,在里面加上這樣一句話:
      clsIme.SetIme(this);

方法二:
使用繼承的方法。
      首先,建立一個獨立的類如下:
      

 1using System;
 2using System.Collections.Generic;
 3using System.ComponentModel;
 4using System.Data;
 5using System.Collections;
 6using System.Drawing;
 7using System.Text;
 8using System.Windows.Forms;
 9using System.Runtime.InteropServices;
10
11namespace MyDemo
12{
13    public class ImeForm:System.Windows.Forms.Form 
14    {
15        //聲明一些API函數(shù)
16        [DllImport("imm32.dll")]
17        public static extern IntPtr ImmGetContext(IntPtr hwnd);
18        [DllImport("imm32.dll")]
19        public static extern bool ImmGetOpenStatus(IntPtr himc);
20        [DllImport("imm32.dll")]
21        public static extern bool ImmSetOpenStatus(IntPtr himc, bool b);
22        [DllImport("imm32.dll")]
23        public static extern bool ImmGetConversionStatus(IntPtr himc, ref int lpdw, ref int lpdw2);
24        [DllImport("imm32.dll")]
25        public static extern int ImmSimulateHotKey(IntPtr hwnd, int lngHotkey);
26        private  const int IME_CMODE_FULLSHAPE = 0x8;
27        private  const int IME_CHOTKEY_SHAPE_TOGGLE = 0x11;
28        //重載Form的OnActivated
29        protected override void OnActivated(EventArgs e)
30        {              
                     base.onActivated(e);           
31            IntPtr HIme = ImmGetContext(this.Handle);
32            if (ImmGetOpenStatus(HIme))  //如果輸入法處于打開狀態(tài)
33            {
34                int iMode = 0;
35                int iSentence = 0;
36                bool bSuccess = ImmGetConversionStatus(HIme, ref iMode, ref iSentence);  //檢索輸入法信息
37                if (bSuccess)
38                {
39                    if ((iMode & IME_CMODE_FULLSHAPE) > 0)   //如果是全角
40                        ImmSimulateHotKey(this.Handle, IME_CHOTKEY_SHAPE_TOGGLE);  //轉(zhuǎn)換成半角
41                }

42
43            }

44        }
        
45    }

46}

47


      使用這個類的方法為:
      修改所有的Form的繼承關(guān)系,比如,你有這樣的一個Form類:
      public partial class Form1 :Form 
      {
      ...
      }
      那么,把它改成:
      public partial class Form1 :ImeForm
      {
      ...
      }
      相信,這樣的修改會很快,全項目查找替換一下即可。
      記住,如果你的Form是多重繼承下來的,例如:FormC派生于FormB,而FormB又派生于FormA,那么,僅僅需要FormA從imeForm派生即可。
方法二的使用優(yōu)勢是明顯的,把Ime的事件從Form最上一層就截取了,避免了在您的Form中控件的多樣性所帶來的困擾。
      還有,網(wǎng)上有一些說的調(diào)整ImeMode和使用ImeModeChanged方法來解決這個問題,建議你暫時(只是暫時)不要使用,因為修改ImeMode根本不能解決窗口切換時輸入法自動變?nèi)堑膯栴},而且ImeModeChanged是在ImeMode改變的時候才觸發(fā),在用戶手工操作輸入法狀態(tài)改變時(比如按Ctrl+Shift)是不會觸發(fā)的。

關(guān)鍵詞:dotnet

贊助商鏈接: