VS2005中文输入法自动转换为全角的两种解决方法

最近在用VS2005做项目的时候,一直忍受着VS2005输入法自动切换到全角的Bug的作怪,一边等待着微软给我们一个解决的方案。但是,我的项目都要作为产品打包出去了,微软还是闷头不对这个Bug出一个解决方法。怎么办?我可以忍受这个输入法来回切换之苦,可用户体验可不会饶过我们的。弄不好,来个集体罢用,让我们都到微软喝西北风去啊!
总不能就这么交出产品出去吧,只有自己动手了。下面我用两种方法来实现如何避免输入法的这个Bug。
方法一:
Form的Pain和遍历Control的Enter方法。
首先,我们为了使您原有的代码更简洁,我们把所要做的步骤封装到一个单独的类中,类代码如下:
1 using System;
2 using System.Runtime.InteropServices;
3
4 namespace MyDemo
5 {
6publicstaticclassclsIme
7{
8//声明一些API函数
9[DllImport("imm32.dll")]
10publicstaticexternIntPtrImmGetContext(IntPtrhwnd);
11[DllImport("imm32.dll")]
12publicstaticexternboolImmGetOpenStatus(IntPtrhimc);
13[DllImport("imm32.dll")]
14publicstaticexternboolImmSetOpenStatus(IntPtrhimc,boolb);
15[DllImport("imm32.dll")]
16publicstaticexternboolImmGetConversionStatus(IntPtrhimc,refintlpdw,refintlpdw2);
17[DllImport("imm32.dll")]
18publicstaticexternintImmSimulateHotKey(IntPtrhwnd,intlngHotkey);
19publicconstintIME_CMODE_FULLSHAPE=0x8;
20publicconstintIME_CHOTKEY_SHAPE_TOGGLE=0x11;
21//重载SetIme,传入Form
22publicstaticvoidSetIme(Formfrm)
23{
24frm.Paint+=newPaintEventHandler(frm_Paint);
25ChangeAllControl(frm);
26}

27//重载SetIme,传入Control
28publicstaticvoidSetIme(Controlctl)
29{
30ChangeAllControl(ctl);
31}

32//重载SetIme,传入对象句柄
33publicstaticvoidSetIme(IntPtrHandel)
34{
35ChangeControlIme(Handel);
36}

37privatestaticvoidChangeAllControl(Controlctl)
38{
39//在控件的的Enter事件中触发来调整输入法状态
40ctl.Enter+=newEventHandler(ctl_Enter);
41//遍历子控件,使每个控件都用上Enter的委托处理
42foreach(ControlctlChildinctl.Controls)
43ChangeAllControl(ctlChild);
44}

45
46staticvoidfrm_Paint(objectsender,PaintEventArgse)
47{
48/**//*有人问为什么使用Pain事件,而不用Load事件或Activated事件,是基于下列考虑:
49*1、在您的Form中,有些控件可能是运行时动态添加的
50*2、在您的Form中,使用到了非.NET的OCX控件
51*3、Form调用子Form的时候,Activated事件根本不会触发*/

52ChangeControlIme(sender);
53}

54//控件的Enter处理程序
55staticvoidctl_Enter(objectsender,EventArgse)
56{
57ChangeControlIme(sender);
58}

59privatestaticvoidChangeControlIme(objectsender)
60{
61Controlctl=(Control)sender;
62ChangeControlIme(ctl.Handle);
63}

64//下面这个函数才是真正检查输入法的全角半角状态
65privatestaticvoidChangeControlIme(IntPtrh)
66{
67IntPtrHIme=ImmGetContext(h);
68if(ImmGetOpenStatus(HIme))//如果输入法处于打开状态
69{
70intiMode=0;
71intiSentence=0;
72boolbSuccess=ImmGetConversionStatus(HIme,refiMode,refiSentence);//检索输入法信息
73if(bSuccess)
74{
75if((iMode&IME_CMODE_FULLSHAPE)>0)//如果是全角
76ImmSimulateHotKey(h,IME_CHOTKEY_SHAPE_TOGGLE);//转换成半角
77}

78}

79}

80}

81}


有人问为什么使用Pain事件,而不用Load事件或Activated事件,我是基于下列考虑:

1、在您的Form中,有些控件可能是运行时动态添加的
2、在您的Form中,使用到了非.NET的OCX控件
3、Form调用子Form的时候,Activated事件根本不会触发

使用这个类的方法为:
在您的界面中,在Load的时候,在里面加上这样一句话:
clsIme.SetIme(this);

方法二:
使用继承的方法。
首先,建立一个独立的类如下:

1 using System;
2 using System.Collections.Generic;
3 using System.ComponentModel;
4 using System.Data;
5 using System.Collections;
6 using System.Drawing;
7 using System.Text;
8 using System.Windows.Forms;
9 using System.Runtime.InteropServices;
10
11 namespace MyDemo
12 {
13publicclassImeForm:System.Windows.Forms.Form
14{
15//声明一些API函数
16[DllImport("imm32.dll")]
17publicstaticexternIntPtrImmGetContext(IntPtrhwnd);
18[DllImport("imm32.dll")]
19publicstaticexternboolImmGetOpenStatus(IntPtrhimc);
20[DllImport("imm32.dll")]
21publicstaticexternboolImmSetOpenStatus(IntPtrhimc,boolb);
22[DllImport("imm32.dll")]
23publicstaticexternboolImmGetConversionStatus(IntPtrhimc,refintlpdw,refintlpdw2);
24[DllImport("imm32.dll")]
25publicstaticexternintImmSimulateHotKey(IntPtrhwnd,intlngHotkey);
26privateconstintIME_CMODE_FULLSHAPE=0x8;
27privateconstintIME_CHOTKEY_SHAPE_TOGGLE=0x11;
28//重载Form的OnActivated
29protectedoverridevoidOnActivated(EventArgse)
30{
base.onActivated(e);
31IntPtrHIme=ImmGetContext(this.Handle);
32if(ImmGetOpenStatus(HIme))//如果输入法处于打开状态
33{
34intiMode=0;
35intiSentence=0;
36boolbSuccess=ImmGetConversionStatus(HIme,refiMode,refiSentence);//检索输入法信息
37if(bSuccess)
38{
39if((iMode&IME_CMODE_FULLSHAPE)>0)//如果是全角
40ImmSimulateHotKey(this.Handle,IME_CHOTKEY_SHAPE_TOGGLE);//转换成半角
41}

42
43}

44}

45}

46}

47


使用这个类的方法为:
修改所有的Form的继承关系,比如,你有这样的一个Form类:
public partial class Form1 :Form
{
...
}
那么,把它改成:
public partial class Form1 :ImeForm
{
...
}
相信,这样的修改会很快,全项目查找替换一下即可。
记住,如果你的Form是多重继承下来的,例如:FormC派生于FormB,而FormB又派生于FormA,那么,仅仅需要FormA从imeForm派生即可。
方法二的使用优势是明显的,把Ime的事件从Form最上一层就截取了,避免了在您的Form中控件的多样性所带来的困扰。
还有,网上有一些说的调整ImeMode和使用ImeModeChanged方法来解决这个问题,建议你暂时(只是暂时)不要使用,因为修改ImeMode根本不能解决窗口切换时输入法自动变全角的问题,而且ImeModeChanged是在ImeMode改变的时候才触发,在用户手工操作输入法状态改变时(比如按Ctrl+Shift)是不会触发的。




PS:最近微软出了补丁。详情请见 这里

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值