10

我在 C# .net 3.5 中创建了自己的单行文本框控件

我已经完全从头开始编写了这个,包括我自己的选择代码、滚动、复制/粘贴等。它甚至可以选择从左到右的文本,而不是我曾经使用过的......

我遇到的问题是 IME(输入法编辑器)集成。IME 在一定程度上确实有效,但我有一些问题。我将在这里列出其中之一。

通常,使用 TextBox,IME 构造在 TextBox 内呈现。但是,在我的控件中,IME 显示在屏幕的右下方。我尝试查看其他自定义控件,包括 SharpDevelop 中使用的那个。然而,这也有同样的问题。输入中文、日文或韩文会导致在屏幕右下角输入 IME。

我需要一些关于如何在我自己的文本框中完成此操作的指示,并且我查看了 IME 的 Windows API,但功能太多,我不知道从哪里开始。

那么,有什么想法吗?

如果任何代码示例在 C++(托管/非托管)或 VB.net 中,那很好。我可以很容易地移动它。

4

1 回答 1

11

好的,看起来 StackOverflow 上的 IME 专家并不多……或者没有人感兴趣。

无论如何,我想通了。

基本上,我必须捕获以下 Windows 消息:

WM_INPUTLANGCHANGE = 0x51
WM_KEYUP = 0x101
WM_CHAR = 0x102
WM_CONVERTREQUESTEX = 0x108
WM_IME_STARTCOMPOSITION = 0x10D
WM_IME_ENDCOMPOSITION = 0x10E
WM_IME_COMPOSITION = 0x10F
WM_IME_SETCONTEXT = 0x281
WM_IME_NOTIFY = 0x282
WM_IME_CONTROL = 0x283
WM_IME_COMPOSITIONFULL = 0x284
WM_IME_SELECT = 0x285
WM_IME_CHAR = 0x286
WM_IME_REQUEST = 0x0288
WM_IME_KEYDOWN = 0x290
WM_IME_KEYUP = 0x291

我正在陷印WM_KEYUPWM_CHAR,因为如果我在合成韩文字符的过程中单击某处,我不会收到合成消息,但我仍然需要将该字符添加到我的文本框中。这是一个奇怪的行为,我想知道它是否是一个错误。

一旦发生这种情况,韩国人、中国人和日本人之间的行为就会不同。

韩语非常简单(不过我不确定汉字转换,因为我不知道如何使用它)。

基本上,对于所有语言,每当我得到 a 时WM_IME_COMPOSITION,我都必须调用ImmGetCompositionStringImm32.dll,正如我在回答这个问题时所描述的那样。然后我将其显示为正在进行的合成,但不要将其添加到我存储的文本中。

组合字符串后,来自 Windows 的消息对于每个 IME 都是不同的。每次,我都可以从WM_IME_COMPOSITION消息中得到它。

在韩语中,LParam将只是GCS_RESULTSTR,并且WParam将是输入的字符,我可以将其转换为char

在日语中,“LParam”将是GCS_RESULTREADSTR | GCS_RESULTREADCLAUSE | GCS_RESULTSTR0 | GCS_RESULTCLAUSE. 我必须使用ImmGetCompositionString从之前的 WM_IME_COMPOSITION 消息中存储的结果,因为此时它将是一个空字符串。

在中文中,LParam将是GCS_RESULTREADCLAUSE | GCS_RESULTSTR0 | GCS_RESULTCLAUSE。它与日语相同,除非先前存储ImmGetCompositionString的内容为空,在这种情况下我需要转换WParam为字符。

在所有这三种情况下,我都必须确保清除显示的正在进行的组合。

如果收到WM_IME_STARTCOMPOSITION,我会设置一个合成标志(并显示正在进行的合成字符串)

如果我收到WM_IME_ENDCOMPOSITION我清除该标志(并清除正在进行的合成字符串)。

有时我没有收到WM_IME_ENDCOMPOSITION,所以我在收到时清除我的标志WM_CHAR


总而言之,这是一次非常有趣的学习体验,而且还在进行中——但 IME 现在终于可以在我的控制下使用了!我一直工作到凌晨 1 点才完成。

于 2010-07-21T01:43:02.167 回答