总不能就这么交出产品出去吧,只有自己动手了。下面我用两种方法来实现如何避免输入法的这个Bug。
方法一:
Form的Pain和遍历Control的Enter方法。
首先,我们为了使您原有的代码更简洁,我们把所要做的步骤封装到一个单独的类中,类代码如下:
using
System;2
using
System.Runtime.InteropServices;3

4
namespace
MyDemo5

{6
publicstaticclassclsIme7


{8
//声明一些API函数9
[DllImport("imm32.dll")]10
publicstaticexternIntPtrImmGetContext(IntPtrhwnd);11
[DllImport("imm32.dll")]12
publicstaticexternboolImmGetOpenStatus(IntPtrhimc);13
[DllImport("imm32.dll")]14
publicstaticexternboolImmSetOpenStatus(IntPtrhimc,boolb);15
[DllImport("imm32.dll")]16
publicstaticexternboolImmGetConversionStatus(IntPtrhimc,refintlpdw,refintlpdw2);17
[DllImport("imm32.dll")]18
publicstaticexternintImmSimulateHotKey(IntPtrhwnd,intlngHotkey);19
publicconstintIME_CMODE_FULLSHAPE=0x8;20
publicconstintIME_CHOTKEY_SHAPE_TOGGLE=0x11;21
//重载SetIme,传入Form22
publicstaticvoidSetIme(Formfrm)23


{24
frm.Paint+=newPaintEventHandler(frm_Paint);25
ChangeAllControl(frm);26
}27
//重载SetIme,传入Control28
publicstaticvoidSetIme(Controlctl)29


{30
ChangeAllControl(ctl);31
}32
//重载SetIme,传入对象句柄33
publicstaticvoidSetIme(IntPtrHandel)34


{35
ChangeControlIme(Handel);36
}37
privatestaticvoidChangeAllControl(Controlctl)38


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

46
staticvoidfrm_Paint(objectsender,PaintEventArgse)47


{48

/**//*有人问为什么使用Pain事件,而不用Load事件或Activated事件,是基于下列考虑:49
*1、在您的Form中,有些控件可能是运行时动态添加的50
*2、在您的Form中,使用到了非.NET的OCX控件51
*3、Form调用子Form的时候,Activated事件根本不会触发*/52
ChangeControlIme(sender);53
}54
//控件的Enter处理程序55
staticvoidctl_Enter(objectsender,EventArgse)56


{57
ChangeControlIme(sender);58
}59
privatestaticvoidChangeControlIme(objectsender)60


{61
Controlctl=(Control)sender;62
ChangeControlIme(ctl.Handle);63
}64
//下面这个函数才是真正检查输入法的全角半角状态65
privatestaticvoidChangeControlIme(IntPtrh)66


{67
IntPtrHIme=ImmGetContext(h);68
if(ImmGetOpenStatus(HIme))//如果输入法处于打开状态69


{70
intiMode=0;71
intiSentence=0;72
boolbSuccess=ImmGetConversionStatus(HIme,refiMode,refiSentence);//检索输入法信息73
if(bSuccess)74


{75
if((iMode&IME_CMODE_FULLSHAPE)>0)//如果是全角76
ImmSimulateHotKey(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);
方法二:
使用继承的方法。
首先,建立一个独立的类如下:
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
MyDemo12

{13
publicclassImeForm:System.Windows.Forms.Form14


{15
//声明一些API函数16
[DllImport("imm32.dll")]17
publicstaticexternIntPtrImmGetContext(IntPtrhwnd);18
[DllImport("imm32.dll")]19
publicstaticexternboolImmGetOpenStatus(IntPtrhimc);20
[DllImport("imm32.dll")]21
publicstaticexternboolImmSetOpenStatus(IntPtrhimc,boolb);22
[DllImport("imm32.dll")]23
publicstaticexternboolImmGetConversionStatus(IntPtrhimc,refintlpdw,refintlpdw2);24
[DllImport("imm32.dll")]25
publicstaticexternintImmSimulateHotKey(IntPtrhwnd,intlngHotkey);26
privateconstintIME_CMODE_FULLSHAPE=0x8;27
privateconstintIME_CHOTKEY_SHAPE_TOGGLE=0x11;28
//重载Form的OnActivated29
protectedoverridevoidOnActivated(EventArgse)30


{base.onActivated(e);
31
IntPtrHIme=ImmGetContext(this.Handle);32
if(ImmGetOpenStatus(HIme))//如果输入法处于打开状态33


{34
intiMode=0;35
intiSentence=0;36
boolbSuccess=ImmGetConversionStatus(HIme,refiMode,refiSentence);//检索输入法信息37
if(bSuccess)38


{39
if((iMode&IME_CMODE_FULLSHAPE)>0)//如果是全角40
ImmSimulateHotKey(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:最近微软出了补丁。详情请见 这里。
2846

被折叠的 条评论
为什么被折叠?



