我可能完全想像得出这一点,但是我可能发誓,有一种方法可以使RichTextBox中的各个Run(或Parapgraph)元素变为只读。我也可能发誓,几周前我尝试了一种方法来自己执行此操作,并对结果感到满意-我隐约记得它看起来像这样:

<RichTextBox x:Name="richTextBox"
             AcceptsTab="True"
             AcceptsReturn="True"
             FontFamily="Courier New"
             FontSize="14">
    <FlowDocument>
        <Paragraph>
            <Run IsReadOnly="True">I wish this was read-only!</Run>
        </Paragraph>
    </FlowDocument>
</RichTextBox>

现在,几周后,我去尝试在RichTextBox中将Run元素设为只读,只是发现似乎不可能。

This post on the MSDN forums似乎证实了这一点。

我完全想象得到了吗?还是有一种方法可以做我想做的事?

最佳答案

好了,我想出了一种适用于我的情况的解决方案-但可能不适用于每个想要这样的人的解决方案。杂乱无章,但确实可以做到。

几天之内,我不会接受自己的答案,以防万一其他人有更好的方法来完成此任务。

首先,我们开始XAML:

<Window x:Class="WpfApplication1.Window1"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="Window1"
        Height="500"
        Width="600">
    <DockPanel LastChildFill="True">
        <RichTextBox x:Name="rtb"
                     FontFamily="Courier New"
                     FontSize="14"
                     PreviewKeyDown="rtb_PreviewKeyDown">
            <FlowDocument>
                <Paragraph>
                    <InlineUIContainer Unloaded="InlineUIContainer_Unloaded">
                        <TextBlock FontFamily="Courier New" FontSize="14">This line of text is not editable.</TextBlock>
                    </InlineUIContainer>
                    <Run Foreground="Blue">But this is editable.</Run>
                </Paragraph>
            </FlowDocument>
        </RichTextBox>
    </DockPanel>
</Window>

以及后面的代码:
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;

namespace WpfApplication1
{
    public partial class Window1 : Window
    {
        public Window1()
        {
            InitializeComponent();
        }

        private void InlineUIContainer_Unloaded(object sender, RoutedEventArgs e)
        {
            (sender as InlineUIContainer).Unloaded -= new RoutedEventHandler(InlineUIContainer_Unloaded);

            TextBlock tb = new TextBlock();
            tb.FontFamily = new FontFamily("Courier New");
            tb.FontSize = 14;
            tb.Text = "This line of text is not editable.";

            TextPointer tp = rtb.CaretPosition.GetInsertionPosition(LogicalDirection.Forward);
            InlineUIContainer iuic = new InlineUIContainer(tb, tp);
            iuic.Unloaded += new RoutedEventHandler(InlineUIContainer_Unloaded);
        }

        private void rtb_PreviewKeyDown(object sender, KeyEventArgs e)
        {
            if (e.Key == Key.Enter)
            {
                var newPointer = rtb.Selection.Start.InsertLineBreak();
                rtb.Selection.Select(newPointer, newPointer);

                e.Handled = true;
            }
        }
    }
}

我的解决方案依赖于以下事实:从用户界面中删除InlineUIContainer时,将调用它的Unloaded()方法。那时,我只是将删除的InlineUIContainer重新插入当前插入符的位置。

与任何骇客一样,也有许多缺点。我发现的缺点如下:
  • 我想成为只读的文本需要用InlineUIContainer包装。这是对该解决方案的一点限制。
  • 我必须捕获'Enter'键并手动插入换行符,否则,每次按下Enter键时InlineUIContainer.Unloaded()都会一直触发。不好玩,但适用于我的情况。

  • 这不是一个很好的解决方案,但我认为它将对我有用。就像我说的,我不会将其标记为我自己的问题的答案-希望其他人会有更好的方法来做到这一点。

    09-04 13:44