WPF DataGrid显格式
2020-12-13 14:46
标签:des style http io os ar for sp strong Formatting WPF DataGrid content depending on business logic data is way too difficult. Especially since MSDN is not telling you anything about it. I have spent weeks to figure out how to get the binding right. Let me show you how it is done to save you time and endless searches on the Internet. The container hierarchy of a DataGrid looks like this: A DataGrid contains Note that The binding gets assigned to a WPF needs two source information to make a binding work: Usually, the Source gets inherited from the The business data example is based on some stock taking figures. A stock item looks like this: The sample data: Even connecting a DataGrid with the business data is not trivial. Basically, a The In this article, data gets only read. If the user should be able to edit the data, use an Formatting a whole column is easy. Just set the property, like The binding here is not involved with the formatting, but specifies the content of the cell (i.e., Formatting the rows is special, because there will be many rows. The The In this example, the background of a row depends on the value of the Formatting just a cell instead of the whole row is a challenge. In a text column, the cell has a Example: In our stock grid, the The most complex case is if the cell format does not depend on the cell value, but some other business data. In our example, the quantity of an item should be displayed as strike through if it is obsolete. To achieve this, the See the Zip file for the complete source code of the sample. This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL) WPF DataGrid显格式 标签:des style http io os ar for sp strong 原文地址:http://www.cnblogs.com/qq247039968/p/4066073.htmlGuide to WPF DataGrid formatting using bindings
4.83 (13 votes)
1
2
3
4
5
Rate:
Votes of 3 or less require a comment
Introduction
WPF DataGrid Structure
DataGrid
DataGridRows
DataGridCell
TextBlock
DataGridRow
s which contain DataGridCell
s which contain exactly one TextBlock
, if it is a TextColumn
and in read mode (editing mode uses a TextBox
). Of course, the visual tree is a bit more complicated: DataGridColumn
is not part of the visual tree. Whatever is defined in DataGridColumn
will be applied to all cells of that column. WPF Binding Basics
FrameworkElement
property, which constitutes the target of the binding.
DataContext
of a parent container, often the Window itself. But DataGrid‘s DataContext
cannot be used for the binding of rows and cells, because each row needs to bind to a different business logic object.DataGridColumn
specifies the binding for the value to be displayed in the cell with the DataGridColumn.Binding
property. The DataGrid creates during runtime a binding for every TextBlock.Text
. Unfortunately, the DataGrid
does not support binding for any other property of TextBlock
. If you try to setup a style for the TextBlock
yourself, the binding will most likely fail, because it wouldn‘t know which business object from the ItemsSource
to use.Business data used
public class StockItem {
public string Name { get; set; }
public int Quantity { get; set; }
public bool IsObsolete { get; set; }
}
Name
Quantity
IsObsolete
Many items
100
false
Enough items
10
false
Shortage item
1
false
Item with error
-1
false
Obsolete item
200
true
Connecting a DataGrid with business data
CollectionViewSource
is used to connect the DataGrid with the business data: CollectionViewSource
does the actual data navigation, sorting, filtering, etc.
CollectionViewSource
in Windows.Resource
Window.Resources>
CollectionViewSource x:Key="ItemCollectionViewSource" CollectionViewType="ListCollectionView"/>
/Window.Resources>
mustThe gotcha here is that you
set the
CollectionViewType
. If you don‘t, the GridView will use
BindingListCollectionView
, which does not support sorting. Of course, MSDN does not explain this anywhere.
DataContext
of the DataGrid to the CollectionViewSource
.DataGrid
DataContext="{StaticResource ItemCollectionViewSource}"
ItemsSource="{Binding}"
AutoGenerateColumns="False"
CanUserAddRows="False">
CollectionViewSource
and assign your business data to the Source property//create business data
var itemList = new List
ObservableCollection
. DataGrid Formatting
Formatting a column
Fontweight
directly in the DataGridColumn
:DataGridTextColumn Binding="{Binding Path=Name}" Header="Name" FontWeight="Bold"/>
Text
property of TextBlock
).Formatting complete rows
DataGrid
offers for this purpose the RowStyle
property. This style will be applied to every DataGridRow
.datagrid.rowstyle>
style targettype="DataGridRow">
DatGridRow
has an Item property, which contains the business logic object for that row. The binding for the DataRow
must therefore bind to itself ! The path is a bit surprising, because Item is of type Object
and doesn‘t know any business data properties. But the WPF binding applies a bit of magic and finds the Quantity
property of StockItem
anyway.Quantity
property of the business object. If there are many items in stock, the background should be white, if only few are left, the background should be grey. The QuantityToBackgroundConverter
performs the necessary calculation:class QuantityToBackgroundConverter: IValueConverter {
public object Convert(object value, Type targetType, object parameter, CultureInfo culture) {
if (value is int) {
int quantity = (int)value;
if (quantity>=100) return Brushes.White;
if (quantity>=10) return Brushes.WhiteSmoke;
if (quantity>=0) return Brushes.LightGray;
return Brushes.White; //quantity should not be below 0
}
//value is not an integer. Do not throw an exception
// in the converter, but return something that is obviously wrong
return Brushes.Yellow;
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture) {
throw new NotImplementedException();
}
}
Formatting a cell based on the displayed value
TextBlock
which needs to be styled. To create a Style
for TextBlock
s is easy, but how can the TextBlock
property be bound to the proper business object ? The DataGrid is binding already the Text
property of the TextBlock
. If the styling depends only on the cell value, we can simply use a self binding to this Text property.Quantity
should always be greater than or equal zero. If a quantity is negative, it is an error and should be displayed in red: Setter Property="Foreground"
Value="{Binding
RelativeSource={RelativeSource Self},
Path=Text,
Converter={StaticResource QuantityToForegroundConverter}}" />
Formatting a cell based on business logic data
TextDecorations
property needs to be linked to the business object of that row. Meaning the TextBlock
has to find the parent DataGridRow
. Luckily, binding to a parent visual object can be done with a relative source:Setter Property="TextDecorations"
Value="{Binding RelativeSource={RelativeSource AncestorType={x:Type DataGridRow}},
Path =Item.IsObsolete,
Converter={StaticResource IsObsoleteToTextDecorationsConverter}}" />
public class IsObsoleteToTextDecorationsConverter: IValueConverter {
public object Convert(object value, Type targetType, object parameter, CultureInfo culture) {
if (value is bool) {
if ((bool)value) {
TextDecorationCollection redStrikthroughTextDecoration =
TextDecorations.Strikethrough.CloneCurrentValue();
redStrikthroughTextDecoration[0].Pen = new Pen {Brush=Brushes.Red, Thickness = 3 };
return redStrikthroughTextDecoration;
}
}
return new TextDecorationCollection();
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture) {
throw new NotImplementedException();
}
}
Code
License
Share