代码之家  ›  专栏  ›  技术社区  ›  Rob

只读运行wpf richtextbox中的元素?

  •  11
  • Rob  · 技术社区  · 15 年前

    我可以完全想象这一点,但我可以发誓有一种方法可以使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 似乎证实了这一点。

    我完全想象了吗?还是有什么方法可以做我想做的?

    2 回复  |  直到 11 年前
        1
  •  7
  •   Rob    15 年前

    好吧,我已经想出了一个适合我的解决方案——但可能不适用于所有想要这样东西的人。它很乱,但它确实起作用。

    几天之内,我不会接受我自己的回答,只是为了以防万一有人能更好地完成这个任务。

    首先,我们来看一下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.Unloaded() 每次按下Enter键时都会继续开火。不好玩,但对我的案子有用。

    这不是一个很好的解决方案,但我认为它对我有用。就像我说的,我还不打算把这个标记为我自己问题的答案——希望其他人有更好的方法来解决这个问题。

        2
  •  1
  •   Andro Selva Yannick Chaze    11 年前

    这可以通过处理两个事件来实现:1)OnMouseDown 2)OnPreviewKeyDown

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Windows.Controls;
    using System.Windows;
    using System.Windows.Input;
    
    namespace WpfApplication2
    {
    public class MultiPartTextBox : TextBox
    {
        private string _prefix;
        private string _content;
    
        public string Prefix
        {
            get { return _prefix; }
            set { _prefix = value;
            Text = _prefix;
            }
        }
    
        public string Content
        {
            get { return _content; }
            set { 
                _content = value;
                Text = _prefix + _content;
            }
        }
    
        public MultiPartTextBox() { _prefix = string.Empty; }
    
        protected override void OnMouseDown(MouseButtonEventArgs e)
        {
    
            base.OnMouseDown(e);
            this.SelectionStart = _prefix.Length;
            this.SelectionLength = 0;
        }
    
        //tab In
        protected override void OnGotFocus(RoutedEventArgs e)
        {
            this.SelectionStart = _prefix.Length;
            this.SelectionLength = 0;
            base.OnGotFocus(e);
        }
    
        protected override void OnPreviewKeyDown(KeyEventArgs e)
        {
            if (e.Key == Key.Back 
                || e.Key == Key.Delete
                || e.Key==Key.Left)
            {
                if (CaretIndex <= _prefix.Length)
                {
                    e.Handled = true;
                    return;
                }
            }
            base.OnPreviewKeyDown(e);
        }
      }
      }
    

    在XAML中,我们必须以以下方式处理它:

       xmlns:uc="clr-namespace:WpfApplication2"
    
           <uc:MultiPartTextBox Height="30" HorizontalAlignment="Left" 
                 Margin="80,94,0,0" x:Name="multiPartTxt1" VerticalAlignment="Top" 
                 Width="224" Prefix="NON-EDITABLE" CaretIndex="4" >            
           </uc:MultiPartTextBox>