WPF: Рисование сверху TextBlock

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

   public class DerivedTextBlock : TextBlock {

      public Boolean DrawExtra {
         get { return (Boolean)GetValue(DrawExtraProperty); }
         set { SetValue(DrawExtraProperty, value); }
      }

      // Using a DependencyProperty as the backing store for DrawExtra.  This enables animation, styling, binding, etc...
      public static readonly DependencyProperty DrawExtraProperty =
          DependencyProperty.Register("DrawExtra", typeof(Boolean), typeof(DerivedTextBlock), new FrameworkPropertyMetadata(false, FrameworkPropertyMetadataOptions.AffectsRender | FrameworkPropertyMetadataOptions.AffectsArrange));

      public DrawingVisual DrawingVisual { get; set; }

      public DerivedTextBlock() {
         DrawingVisual = this.CreateDrawingVisualRectangle();
      }

      protected override int VisualChildrenCount {
         get {
            //if we want to draw our extra info, add one to
            // our visualChildrenCount, usually with a textblock it is 0
            if (DrawExtra) {
               return base.VisualChildrenCount + 1;
            }
            else {
               return base.VisualChildrenCount;
            }
         }
      }

      protected override Visual GetVisualChild(int index) {
         return DrawingVisual;
      }

      // Create a DrawingVisual that contains a rectangle.
      private DrawingVisual CreateDrawingVisualRectangle() {
         DrawingVisual drawingVisual = new DrawingVisual();

         // Retrieve the DrawingContext in order to create new drawing content.
         DrawingContext drawingContext = drawingVisual.RenderOpen();

         // Create a rectangle and draw it in the DrawingContext.
         Rect rect = new Rect(new Point(10.0, 0), new Size(10.0 / 2.0, 10));
         drawingContext.DrawRectangle(Brushes.LightBlue, (Pen)null, rect);

         // Persist the drawing content.
         drawingContext.Close();

         return drawingVisual;
      }
   }

Причина я хочу сделать это: у Нас есть datagrid с большим количеством ячеек, каждого текста отображения ячейки. мы показываем некоторую информацию о проверке о ячейках, и мы делаем это при помощи шаблона с textblock и некоторыми путями hosten в сетке. издержки этого добавляют дополнительные элементы к визуальному дереву и когда мы должны перерисовать (на загрузке, переключая окна или на виде), требуется намного дольше больше элементов в визуальном дереве. когда это - просто textblock, это о 1/3 - 1/2 быстрее, чем иметь контроль с сеткой. Таким образом, мы хотели бы потянуть наш материал проверки прямо сверху текстового поля.

1
задан Aran Mulholland 24 June 2010 в 03:22
поделиться

1 ответ

Ваши проблемы:

  1. GetVisualChild () должен возвращать base.GetVisualChild (index), кроме случаев, когда index == base.VisualChildrenCount.
  2. Вы забыли вызвать AddVisualChild (), когда DrawingExtra становится истинным или изменяет DrawingVisual
  3. Вы забыли вызвать RemoveVisualChild (), когда DrawingExtra становится ложным или изменяется DrawingVisual

Вы можете исправить # 2 и # 3, установив PropertyChangedCallback на DrawingExtra и добавляем код в установщик DrawingVisual.

Объяснение: Это вызов AddVisualChild (), который фактически добавляет визуал в дерево. Происходит то, что ваш визуальный элемент обнаруживается и отображается «случайно» из-за вашей ошибки в GetVisualChild (), но он не связывается должным образом с визуальным деревом, поэтому вы столкнетесь со многими проблемами.

Обновление

Я отредактировал ваш код, как описано выше, и он отлично сработал. Вот изменения:

...
      {
        PropertyChangedCallback = (obj, e) =>
          {
            var textBlock = (DerivedTextBlock)obj;
            if((bool)e.OldValue) textBlock.RemoveVisualChild(textBlock.DrawingVisual);
            if((bool)e.NewValue) textBlock.AddVisualChild(textBlock.DrawingVisual);
          }
      });

  public DrawingVisual DrawingVisual
  {
    get { return _drawingVisual; }
    set
    {
      if(DrawExtra) RemoveVisualChild(_drawingVisual);
      _drawingVisual = value;
      if(DrawExtra) AddVisualChild(_drawingVisual);
    }
  }
  private DrawingVisual _drawingVisual;

...

  protected override int VisualChildrenCount
  {
    get { return base.VisualChildrenCount + (DrawExtra ? 1 : 0); }
  }

  protected override Visual GetVisualChild(int index)
  {
    return index==base.VisualChildrenCount ? DrawingVisual : base.GetVisualChild(index);
  }
2
ответ дан 2 September 2019 в 23:32
поделиться
Другие вопросы по тегам:

Похожие вопросы: