突出显示搜索TextBlock

问题描述:

我的目标是创建一个具有新的依赖项属性SearchText的自定义TextBlock控件。该属性将包含一个正则表达式。在TextBlock文本中出现的所有正则表达式将使用自定义样式(另一个DP)高亮显示。突出显示搜索TextBlock

我当前的实现包括清除TextBlock的InlineCollection中的所有Inline对象。然后,我为TextBlock填充带有未着色文本的运行,并针对应用了样式的突出显示的文本运行(此方法不支持直接向TextBlock添加内联,而是必须使用TextBlock.TextProperty)。

很好用,但有时在尝试清除Inline时出现一个奇怪的异常:InvalidOperationException:“此时由于树正在进行,无法修改此节点的逻辑子节点。

这个问题似乎与this之一有关。我正在修改TextChanged函数中的内联,但我使用一个标志来避免无限递归编辑。

有关如何构建此自定义控件的任何想法?有一个更好的方法吗?我如何解决这个异常?

谢谢!

仍然不确定是否有更好的方法来完成此操作,但我似乎找到了解决办法。

我正在更新由TextProperty和SearchTextProperty的更改通知触发的函数中的内联函数/运行。

现在我使用DispatcherPriority.Normal从更改通知中的Dispatcher.BeginInvoke()调用中触发高亮/更新代码。

如果有人想的是如何做到这一点的例子,我发现this

+0

建议提供一个链接内容摘要,以防链接中断。 – 2012-10-26 20:53:46

在我的实现,我只需添加另一个依赖属性解决了这个名为OriginalText。修改后,我更新了Text属性并更新了突出显示。代码如下:

public class HighlightTextBlock : TextBlock 
{ 
    public string HighlightedText 
    { 
     get { return (string)GetValue(HighlightedTextProperty); } 
     set { SetValue(HighlightedTextProperty, value); } 
    } 

    public static readonly DependencyProperty HighlightedTextProperty = 
     DependencyProperty.Register("HighlightedText", typeof(string), typeof(HighlightTextBlock), new UIPropertyMetadata(string.Empty, UpdateHighlightEffect)); 

    public static readonly DependencyProperty OriginalTextProperty = DependencyProperty.Register(
     "OriginalText", typeof(string), typeof(HighlightTextBlock), new PropertyMetadata(default(string), OnOriginalTextChanged)); 

    private static void OnOriginalTextChanged(DependencyObject obj, DependencyPropertyChangedEventArgs args) 
    { 
     var block = ((HighlightTextBlock)obj); 
     block.Text = block.OriginalText; 
     block.UpdateHighlightEffect(); 
    } 

    public string OriginalText 
    { 
     get { return (string)GetValue(OriginalTextProperty); } 
     set { SetValue(OriginalTextProperty, value); } 
    } 

    private static void UpdateHighlightEffect(DependencyObject sender, DependencyPropertyChangedEventArgs e) 
    { 
     if (!(string.IsNullOrEmpty(e.NewValue as string) && string.IsNullOrEmpty(e.OldValue as string))) 
      ((HighlightTextBlock)sender).UpdateHighlightEffect(); 
    } 

    private void UpdateHighlightEffect() 
    { 
     if (string.IsNullOrEmpty(HighlightedText)) return; 

     var allText = GetCompleteText(); 

     Inlines.Clear(); 

     var indexOfHighlightString = allText.IndexOf(HighlightedText, StringComparison.InvariantCultureIgnoreCase); 

     if (indexOfHighlightString < 0) 
     { 
      Inlines.Add(allText); 
     } 
     else 
     { 
      Inlines.Add(allText.Substring(0, indexOfHighlightString)); 
      Inlines.Add(new Run() 
          { 
           Text = allText.Substring(indexOfHighlightString, HighlightedText.Length), 
           Background = Consts.SearchHighlightColor, 
          }); 
      Inlines.Add(allText.Substring(indexOfHighlightString + HighlightedText.Length)); 
     } 

    } 

    private string GetCompleteText() 
    { 
     var allText = Inlines.OfType<Run>().Aggregate(new StringBuilder(), (sb, run) => sb.Append(run.Text), sb => sb.ToString()); 
     return allText; 
    } 
} 
+0

这太好了。感谢您发布您的解决方案。 – 2015-06-26 12:53:09