Только для чтения. Выполнить элементы в WPF RichTextBox?

Я могу полностью представить это, но я мог бы поклясться, что есть способ сделать отдельные элементы Run (или Parapgraph) в RichTextBox только для чтения. Я также мог поклясться, что попробовал метод для этого сделать несколько недель назад и был доволен результатами - я смутно помню, что это выглядело примерно так:

<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>

Теперь, несколько недель спустя, я пытаюсь сделать элементы Run доступными только для чтения в RichTextBox, только чтобы найти его, похоже, не представляется возможным.

Этот пост на форумах MSDN, похоже, подтверждает это.

Я полностью это себе представлял? Или есть способ сделать то, что я хочу сделать?

Ответ 1

Хорошо, я придумал решение, которое работает для моего дела, но может не работать для всех, кто хочет чего-то подобного. Это беспорядочно, но он выполняет эту работу.

Я не буду принимать свой собственный ответ в течение нескольких дней, на случай, если у кого-то есть лучший способ выполнить это.

Здесь мы идем, во-первых, 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;
            }
        }
    }
}

Мое решение зависит от того, что, когда a InlineUIContainer удаляется из пользовательского интерфейса, вызывается метод Unloaded(). В этот момент я просто вставляю удаленный InlineUIContainer в текущую позицию каретки.

Как и при любом взломе, существует множество недостатков. Недостатки, которые я нахожу, заключаются в следующем:

  • Текст, который я хочу читать только для чтения, должен быть обернут в InlineUIContainer. Это немного ограничивает это решение.
  • Я должен захватить ключ 'Enter' и вставить разрывы строк вручную, в противном случае InlineUIContainer.Unloaded() продолжает стрелять каждый раз, когда нажимается клавиша Enter. Не весело, но это работает для моего дела.

Это не отличное решение, но я думаю, что это сработает для меня. Как я уже сказал, я не стану отмечать это как ответ на свой вопрос, но, надеюсь, у кого-то еще будет лучший способ сделать это.

Ответ 2

Это может быть достигнуто путем обработки двух событий: 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>