我可能完全是在想象这一点,但我可以发誓有一种方法可以使 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 元素设置为只读,却发现这似乎不可能。
MSDN 论坛上的这篇文章 http://social.msdn.microsoft.com/forums/en-US/wpf/thread/329079fc-8555-4192-b90e-6fccd43bf28a似乎证实了这一点。
这是我完全想象出来的吗?或者有办法做我想做的事吗?
好吧,我已经想出了一个适合我的情况的解决方案 - 但可能不适用于每个想要这样的东西的人。虽然很乱,但它能完成工作。
几天内我不会接受我自己的答案,以防万一其他人有更好的方法来完成这个任务。
首先,我们来看看 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
已从 UI 中删除,它是Unloaded()
方法被调用。此时,我只需重新插入已删除的InlineUIContainer
在当前插入符位置。
与任何黑客一样,也有很多缺点。我发现的缺点如下:
- 我想要只读的文本需要包裹在
InlineUIContainer
。这对该解决方案有一点限制。
- 我必须捕获“Enter”键并手动插入换行符,否则,
InlineUIContainer.Unloaded()
每次按下 Enter 键时都会持续触发。不好玩,但它适用于我的情况。
这不是一个很好的解决方案,但我认为它对我有用。就像我说的,我不会将此标记为我自己问题的答案 - 希望其他人有更好的方法来做到这一点。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)