WPF DataGrid Динамическое выделение ячеек ширины

У меня есть приложение, которое использует WPF Data Grid. Эта сетка представляет собой набор результатов испытаний. Если результат теста выходит за пределы минимального и максимального допустимых значений, я хочу выделить эту ячейку красным цветом. В настоящее время у меня это работает, но я не совсем доволен выделением.

Вот как это выглядит сейчас:

введите описание изображения здесь

Вот желаемый вид (через некоторое изображение):

введите описание изображения здесь

Обратите внимание, что выделение в первом примере занимает всю ширину ячейки. Я надеюсь на желаемый пример, где он потребляет столько же места, сколько и самый широкий результат с небольшим запасом с обеих сторон. Имейте в виду, что результат в любой ячейке может варьироваться от 0 до 1920К от одного образца к другому. Это крайний случай, но я хочу, чтобы выделенная область увеличивалась и уменьшалась в результате.

Только к вашему сведению, эти результаты обновляются в настраиваемом таймере, который срабатывает где-то между 10 мс и 10 секунд в зависимости от конфигурации пользователя.

Ниже приведен код, который генерирует первый пример (извините за большой объем кода). Интересными битами являются DataGridCellStyle, ResultCellStyle и CellTemplate.

XAML

<Window x:Class="Stackru_HighlightCell.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:local="clr-namespace:Stackru_HighlightCell"
    mc:Ignorable="d"
    Loaded="Window_Loaded"
    Title="MainWindow" Height="350" Width="525">
<Window.Resources>
    <Style TargetType="DataGridColumnHeader">
        <Setter Property="Background" Value="Transparent" />
        <Setter Property="Foreground" Value="White" />
        <Setter Property="BorderThickness" Value="0" />
        <Setter Property="HorizontalContentAlignment" Value="Center" />
    </Style>

    <ControlTemplate x:Key="CellTemplate" TargetType="{x:Type DataGridCell}">
        <Border Background="{TemplateBinding Background}">
            <ContentPresenter Margin="12,0,0,0" />
        </Border>
    </ControlTemplate>

    <Style x:Key="DataGridCellStyle" TargetType="DataGridCell">
        <Setter Property="Background" Value="#707070" />
        <Setter Property="BorderThickness" Value="0" />
        <Setter Property="HorizontalContentAlignment" Value="Left" />
        <Style.Triggers>
            <Trigger Property="IsSelected" Value="True">
                <Setter Property="Foreground" Value="#CCCCCC" />
            </Trigger>
        </Style.Triggers>
    </Style>

    <Style x:Key="ResultCellStyle" TargetType="DataGridCell" 
           BasedOn="{StaticResource DataGridCellStyle}">
        <Setter Property="Template" Value="{StaticResource CellTemplate}" />
        <Style.Triggers>
            <DataTrigger Binding="{Binding Path=IsResultOutOfBounds, 
                                           StringFormat={}{0:0.00}}" 
                                           Value="True">
                <Setter Property="Background" Value="Red" />
                <Setter Property="Foreground" Value="White" />
            </DataTrigger>
        </Style.Triggers>
    </Style>

    <Style TargetType="DataGrid" BasedOn="{x:Null}">
        <Setter Property="RowBackground" Value="#707070" />
        <Setter Property="AutoGenerateColumns" Value="False" />
        <Setter Property="IsReadOnly" Value="True" />
        <Setter Property="Background" Value="#666666" />
        <Setter Property="GridLinesVisibility" Value="None" />
        <Setter Property="BorderBrush" Value="Transparent" />
        <Setter Property="BorderThickness" Value="0" />
        <Setter Property="CanUserSortColumns" Value="False" />
        <Setter Property="HeadersVisibility" Value="Column" />
        <Setter Property="HorizontalAlignment" Value="Stretch" />
        <Setter Property="Foreground" Value="#CCCCCC" />
        <Setter Property="RowDetailsVisibilityMode" Value="Collapsed" />

        <Setter Property="CellStyle" Value="{StaticResource DataGridCellStyle}" />
    </Style>

    <!-- A left justified DataGridTextColumn -->
    <Style x:Key="ElementLeftJustified">
        <Setter Property="TextBlock.HorizontalAlignment" Value="Left" />
        <Setter Property="TextBlock.Margin" Value="15,0,5,0" />
    </Style>

    <!-- A right justified DataGridTextColumn -->
    <Style x:Key="ElementRightJustified">
        <Setter Property="TextBlock.HorizontalAlignment" Value="Right" />
        <Setter Property="TextBlock.Margin" Value="0,0,5,0" />
    </Style>

</Window.Resources>

<Grid Background="#FF666666">
    <Border Margin="20" >
        <DataGrid x:Name="_testSummaryGrid"
              ItemsSource="{Binding TestResults}">

            <DataGrid.Columns>
                <DataGridTextColumn 
                    MinWidth="75" Header="Test"
                    Binding="{Binding TestName}" 
                    ElementStyle="{StaticResource ElementLeftJustified}" />

                <DataGridTextColumn 
                    MinWidth="75" Header="Min" 
                    Binding="{Binding Min, StringFormat={}{0:0.00}}" 
                    ElementStyle="{StaticResource ElementRightJustified}" />


                <DataGridTextColumn 
                    MinWidth="75" Header="Result" 
                    Binding="{Binding Result, StringFormat={}{0:0.00}}" 
                    ElementStyle="{StaticResource ElementRightJustified}" 
                    CellStyle="{StaticResource ResultCellStyle}" />

                <DataGridTextColumn 
                    MinWidth="75" Header="Max" 
                    Binding="{Binding Max, StringFormat={}{0:0.00}}" 
                    ElementStyle="{StaticResource ElementRightJustified}" />
            </DataGrid.Columns>
        </DataGrid>
    </Border>

</Grid>
</Window>

Модель представления

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.ComponentModel;
using System.Collections.ObjectModel;

namespace Stackru_HighlightCell
{
    public class ResultsViewModel : ViewModelBase
    {
        public ResultsViewModel()
        {
            TestResults = new ObservableCollection<TestResult>
            {
            { new TestResult { TestGroup = "Circle",TestName = "Radius",
                               Min = 100, Max = 153, Result = 150} },
            { new TestResult { TestGroup = "Circle", TestName = "Min Radius",
                               Min = 0, Max = 90, Result = 97.59 } },
            // And so on ...
            };
        }

        public ObservableCollection<TestResult> TestResults { get; set; }

    }

    public class TestResult : ViewModelBase
    {
        public string TestGroup { get; set; }
        public string TestName { get; set; }
        public double Result { get; set; }
        public double Min { get; set; }
        public double Max { get; set; }
        public bool IsResultOutOfBounds { get { return !(Result >= Min && Result <= Max); } }

    }

    public class ViewModelBase : INotifyPropertyChanged
    {
        public event PropertyChangedEventHandler PropertyChanged;
        private void FirePropertyChanged(string property)
        {
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(property));
        }
    }
}

1 ответ

Решение

Я думаю, что это должно делать то, что вы хотите:

<ControlTemplate x:Key="ResultCellTemplate" TargetType="{x:Type DataGridCell}">
    <Border Background="{TemplateBinding Background}">
        <Grid 
            Margin="12,0,0,0"
            HorizontalAlignment="Right"
            >
            <Grid.ColumnDefinitions>
                <ColumnDefinition Width="Auto" SharedSizeGroup="Result" />
            </Grid.ColumnDefinitions>
            <Border Grid.Column="0" x:Name="ContentPresenterBorder">
                <ContentPresenter 
                    />
            </Border>
        </Grid>
    </Border>

    <ControlTemplate.Triggers>
        <!-- 
        That stringformat you had will have been ignored because the target
        type isn't string. 
        -->
        <DataTrigger Binding="{Binding IsResultOutOfBounds}" Value="True">
            <Setter TargetName="ContentPresenterBorder" Property="Background" Value="Red" />
            <Setter TargetName="ContentPresenterBorder" Property="TextElement.Foreground" Value="White" />
        </DataTrigger>
    </ControlTemplate.Triggers>
</ControlTemplate>

<Style x:Key="ResultCellStyle" TargetType="DataGridCell" 
    BasedOn="{StaticResource DataGridCellStyle}">
    <Setter Property="Template" Value="{StaticResource ResultCellTemplate}" />
</Style>

... но не забудьте установить Grid.IsSharedSizeScope="True" на DataGrid, Это работает с SharedSizeGroup="Result" на ColumnDefinition чтобы убедиться, что любой столбец сетки с именем "Результат" в любом месте в пределах DataGrid будет динамически измеряться до той же ширины.

<DataGrid 
    x:Name="_testSummaryGrid"
    ItemsSource="{Binding TestResults}"
    Grid.IsSharedSizeScope="True"
    >

Отличный пример, кстати. Было бы лучше урезать до двух столбцов DataGrid, но я вставил его, нажал F5, и это сработало. И важную часть было не сложно найти.

Другие вопросы по тегам