首页 > 解决方案 > 为什么我的 RichTextBox 没有更改前景色数据绑定

问题描述

所以我目前正在尝试将消息从进程打印到richtextbox,除了我希望它以红色文本显示错误消息和以绿色显示正常消息之外,一切都很好。

出于某种原因,它都是绿色的,我认为这是因为我正在绑定前景而不是附加到 RTB,但我不确定。

如何正确到达错误消息变为红色而正常消息变为绿色的位置。

主窗口.cs

 public partial class MainWindow : Window
    {
        static Server server = new Server();
        private readonly Thread ConsoleThread = new Thread(() => { server.Start("Start.bat"); });

        public MainWindow()
        {
            InitializeComponent();
            DataContext = server;
        }

        private void ButtonStart(object sender, RoutedEventArgs e)
        {
            ConsoleThread.Start();
        }

        private void tbConsole_TextChanged(object sender, System.Windows.Controls.TextChangedEventArgs e)
            => tbConsole.ScrollToEnd();
    }

服务器.cs

class Server : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;
        private Process pServer = new Process();

        private Brush color;

        public Brush MessageColor
        {
            get { return color; }
            set
            {
                color = value;
                OnPropertyChanged("MessageColor");
            }
        }

        private string outStream;
        public string OutputStream
        {
            get { return outStream; }
            set
            {
                if (!string.IsNullOrEmpty(value))
                {
                    outStream += value + Environment.NewLine;
                    OnPropertyChanged("OutputStream");
                }
            }
        }


        public void Start(string Path)
        {

            pServer.StartInfo.FileName = Path;
            pServer.StartInfo.UseShellExecute = false;
            pServer.StartInfo.RedirectStandardOutput = true;
            pServer.StartInfo.RedirectStandardError = true;


            pServer.OutputDataReceived += OKDataReceived;
            pServer.ErrorDataReceived += ErrorDataReceived;


            pServer.Start();

            pServer.BeginOutputReadLine();
            pServer.BeginErrorReadLine();
        }

        private void OKDataReceived(object sender, DataReceivedEventArgs e)
        {
            MessageColor = Brushes.Green;
            OutputStream = e.Data;
        }

        private void ErrorDataReceived(object sender, DataReceivedEventArgs e)
        {
            MessageColor = Brushes.Red;
            OutputStream = e.Data;
        }

        public void OnPropertyChanged(string propertyName)
        {
            PropertyChangedEventHandler handle = PropertyChanged;
            if (handle != null)
            {
                PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
            }
        }
    }

XAML

 <Grid>
        <Button Click="ButtonStart" Content="Start" Margin="325,63,353,319"/>
        <xctk:RichTextBox FontSize="5" Name="tbConsole" Margin="143,149,164,81" BorderBrush="Gray" Padding="10" ScrollViewer.VerticalScrollBarVisibility="Auto" Text="{Binding Path=OutputStream, Mode=TwoWay}" Foreground="{Binding Path=MessageColor}" />
    </Grid>

标签: c#.netxamlmvvmdata-binding

解决方案


是的,我认为您是对的:该Foreground属性是指 中的所有文本RichTextBox,而不是单个项目。

我可以建议您的问题的解决方案,但请耐心等待...

单独收集消息

当我说“消息”时,我的意思是任何被传递给OKDataReceivedand的东西ErrorDataReceived

您的架构中存在一个问题,即您无法区分流程中的“好”消息和“坏”消息。OutputStream两者都包含,并且当MessageColor更改时,它将更改所有以前的消息,而不仅仅是最新消息。

我建议更改创建一个Message带有消息文本和标志的类,以判断它是“好”消息还是“坏”消息。当您从流程中获取新数据时,您将创建Message并相应地设置“好”或“坏”标志。

然后用OutputStream类型的属性替换ObservableCollection<Message>。从流程创建新Message的后,您可以将其添加到此ObservableCollection<Message>. 这是最终将绑定到您的视图的属性。请注意,当新项目添加到时,ObservableCollection它会自动刷新绑定。

现在,我没有忘记您关于如何在视图上以不同颜色显示文本的原始问题。我可以给你一个建议,但这取决于ObservableCollection<Message>模型中有一个。

将消息转换为 FlowDocument

将 绑定ObservableCollection<Message>RichTextBox控件,但使用 anIValueConverter将所有Messages转换为FlowDocument。每个 Message 都将被转换为一个包含文本的Paragraph带有 a的。Run然后,您可以将ForegroundParagraph 上的属性设置为Message. 您还需要将该属性绑定到该RichTextBox.Document属性。MSDN 文章RichTextBox有一个很好的小例子,我想你可以从那里弄清楚。

这种方法的主要缺点是,每当进程发出新消息时,都需要将整个消息再次ObservableCollection转换为 a FlowDocument,而不仅仅是新项目。但我认为这对性能的影响是微乎其微的。

我认为这将是最好的方法。最后,您将拥有一个带有可区分消息的视图模型,一个可以根据需要修改的转换器,并且视图本身只需进行少量更改。


推荐阅读