3

我正在尝试在 WPF RichTextBox 中启用拼写检查。MSDN 写道System.Windows.Controls.SpellCheck可用于启用 TextBox 和 RichTextBox 控件的拼写检查。

不幸的是,以下代码对我不起作用:

<RichTextBox SpellCheck.IsEnabled="True" xml:lang="en-US"></RichTextBox>

这很奇怪,因为如果我使用纯文本框,它工作得非常好(如果我拼错了一些东西,我可以看到红线)。

不幸的是,到目前为止,我在 SO 上找到的每个 答案都只提到将属性设置SpellCheck.IsEnabled为一种受支持的语言,但我不知道为什么在内置 RichTextBoxes 的情况下这种方法在我的计算机上不起作用?TrueLanguage

更新:

如果我这样写,运行中的文本将带有下划线:

<RichTextBox SpellCheck.IsEnabled="True">
    <FlowDocument Language="en">
         <Paragraph>
             <Run>asdfasdf</Run>
         </Paragraph>
    </FlowDocument>
</RichTextBox>

但不幸的是,如果我尝试输入其他文本,它将被忽略。看起来该属性Language未在编辑的内容上设置为英语。我试图设置甚至Thread's CurrentCulture没有CurrentUICulture结果......

4

1 回答 1

6

好的,我终于找到了解决问题的方法。如果您深入研究 WPF 源代码,则可以很容易地发现问题:有一个名为的内部类TextEditorTyping,它有一个名为的方法DoTextInput,负责插入用户输入字符。SetSelectedText该方法通过调用TextEditorTextEditor是另一个为各种控件提供文本编辑服务的内部类,例如)设置插入范围的区域性属性RichTextBox。这是该DoTextInput方法的一部分:

IDisposable disposable = This.Selection.DeclareChangeBlock();
using (disposable)
{
    ITextSelection selection = This.Selection;
    if (!This.AllowOvertype || !This._OvertypeMode)
    {
         flag = false;
    }
    else
    {
         flag = str != "\t";
    }
    ((ITextRange)selection).ApplyTypingHeuristics(flag);
    // SETTING THE CULTURE ->
    This.SetSelectedText(str, InputLanguageManager.Current.CurrentInputLanguage);
    ITextPointer textPointer = This.Selection.End.CreatePointer(LogicalDirection.Backward);
    This.Selection.SetCaretToPosition(textPointer, LogicalDirection.Backward, true, true);
    undoCloseAction = UndoCloseAction.Commit;
}

因此该方法是使用与InputLanguageManager.Current.CurrentInputLanguageWindows 中当前输入语言相对应的 。如果您使用不同于英语的输入语言(这是FrameworkElement.LanguageProperty的默认值),那么如果您在 RichTextBox 中编辑文本,则 FlowDocument 中插入的元素会将当前输入语言作为其Language属性。例如,如果您的输入语言是匈牙利语 ( hu-hu),您的 FlowDocument 将如下所示:

<FlowDocument>
     <Paragraph>
         <Run xml:lang="hu-hu">asdfasdf</Run>
     </Paragraph>
</FlowDocument>

这个网站描述了同样的问题。

幸运的是,有一个解决方法。我们已经看到了DoTextInput方法的来源,里面有一个 using 块:

IDisposable disposable = This.Selection.DeclareChangeBlock();
using (disposable)
{
    ...
    // SETTING THE CULTURE ->
    This.SetSelectedText(str, InputLanguageManager.Current.CurrentInputLanguage);
    ...
}

这是一个更改块,它在最后一行被释放 - 在它被释放后,TextContainerChanged事件被触发,我们可以通过覆盖以下OnTextChanged方法来处理RichTextBox

protected override void OnTextChanged(TextChangedEventArgs e)
{
    var changeList = e.Changes.ToList();
    if (changeList.Count > 0)
    {
        foreach (var change in changeList)
        {
            TextPointer start = null;
            TextPointer end = null;
            if (change.AddedLength > 0)
            {
                start = this.Document.ContentStart.GetPositionAtOffset(change.Offset);
                end = this.Document.ContentStart.GetPositionAtOffset(change.Offset + change.AddedLength);
            }
            else
            {
                int startOffset = Math.Max(change.Offset - change.RemovedLength, 0);
                start = this.Document.ContentStart.GetPositionAtOffset(startOffset);
                end = this.Document.ContentStart.GetPositionAtOffset(change.Offset);
            }

            if (start != null && end != null)
            {
                var range = new TextRange(start, end);
                range.ApplyPropertyValue(FrameworkElement.LanguageProperty, Document.Language);
            }
        }
    }
    base.OnTextChanged(e);
}

在这里,我们将编辑范围的语言重置为正确的值 - 到Document.Language. 在此解决方法之后,您可以使用 WPF 拼写检查 - 例如,在法语中:

<My:CultureIndependentRichTextBox xml:lang="fr-FR" SpellCheck.IsEnabled="True">
     <FlowDocument>
     </FlowDocument>
</My:CultureIndependentRichTextBox>

它会神奇地起作用。:)

于 2013-09-24T14:40:34.160 回答