My DataGrid
's ItemSource
returns empty objects or null. What can be the case ? All properties are bound. While adding data, it shows proper on the Grid, but while retrieving it gives null.
XML
<DataGrid AutoGenerateColumns="False" Grid.Row="2" Height="208" HorizontalAlignment="Left" Margin="20,71,0,0" Name="dgvWell" VerticalAlignment="Top" Width="528" HorizontalScrollBarVisibility="Visible"
VerticalScrollBarVisibility="Visible" BorderBrush="#FFB7B39D" Background="LightYellow" RowBackground="LightGray" AlternatingRowBackground="#FFFFFFF5" BorderThickness="10" CanUserReorderColumns="False" CanUserSortColumns="False" FontSize="13" CanUserAddRows="False">
<DataGrid.Columns>
<DataGridTextColumn Header="Layer Name" Width="80" Binding="{Binding LayerName}"/>
<DataGridTextColumn Width="100" Binding="{Binding Porosity}">
<DataGridTextColumn.Header>
<Grid Width="100">
<Grid.RowDefinitions>
<RowDefinition />
<RowDefinition/>
</Grid.RowDefinitions>
<TextBlock Text="Porosity" Grid.Row="0"/>
<ComboBox Grid.Row="1" Width="70" HorizontalAlignment="Center" Name="cboPorosity">
<ComboBoxItem Content="pascal" IsSelected="True" />
<ComboBoxItem Content="psi"/>
<ComboBoxItem Content="bar"/>
<ComboBoxItem Content="barye"/>
</ComboBox>
</Grid>
</DataGridTextColumn.Header>
</DataGridTextColumn>
<DataGridTextColumn Header="Permeability" Binding="{Binding Permeability}"/>
<DataGridTextColumn Width="120" Binding="{Binding Path= PerforationStartDepth,Mode=TwoWay}" ClipboardContentBinding="{Binding PerforationStartDepth}">
<DataGridTextColumn.Header>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="25"/>
<RowDefinition/>
</Grid.RowDefinitions>
<TextBlock Text="Perforation Start" Grid.Row="0"/>
<ComboBox Grid.Row="1" Width="60" Name="cboPerfStart">
<ComboBoxItem Content="ft" IsSelected="True" />
<ComboBoxItem Content="M"/>
<ComboBoxItem Content="cm"/>
</ComboBox>
</Grid>
</DataGridTextColumn.Header>
</DataGridTextColumn>
<DataGridTextColumn Width="145" Binding="{Binding PerforationEndDepth}">
<DataGridTextColumn.Header>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="25"/>
<RowDefinition/>
</Grid.RowDefinitions>
<TextBlock Text="Perforation End" Grid.Row="0"/>
<ComboBox Grid.Row="1" Width="60" Name="cboPerfEnd">
<ComboBoxItem Content="ft" IsSelected="True"/>
<ComboBoxItem Content="M"/>
<ComboBoxItem Content="cm"/>
</ComboBox>
</Grid>
</DataGridTextColumn.Header>
</DataGridTextColumn>
<DataGridTextColumn Width="140" Binding="{Binding ReservoirPressure,Mode=TwoWay}">
<DataGridTextColumn.Header>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height="25"/>
<RowDefinition/>
</Grid.RowDefinitions>
<TextBlock Text="Reservoir Pressure" Grid.Row="0"/>
<ComboBox Grid.Row="1" Width="60" Name="cboResPress">
<ComboBoxItem Content="pascal" IsSelected="True" />
<ComboBoxItem Content="psi"/>
<ComboBoxItem Content="bar"/>
<ComboBoxItem Content="barye"/>
</ComboBox>
</Grid>
</DataGridTextColumn.Header>
</DataGridTextColumn>
<DataGridTextColumn Width="145" Header="Water Cut" Binding="{Binding WaterCut}"></DataGridTextColumn>
</DataGrid.Columns>
</DataGrid>
The backend object:
public struct Step2Data
{
public string LayerName { get; set; }
public int Porosity { get; set; }
public int Permeability { get; set; }
public int PerforationStartDepth { get; set; }
public int PerforationEndDepth { get; set; }
public int ReservoirPressure { get; set; }
public int WaterCut { get; set; }
}
List<Step2Data> step2datas = dgvWell.ItemsSource as List<Step2Data>;
The above retrieval returns null.
Can you help me know, why does this grid doesn't return proper ItemSource.
EDIT
Based on your eg which helped me greatly I implemented my code. In my view, I have a combobox that's lets user select a number. Based on that number, that many number of rows get added to the grid. I tried to work on it, but it doesn't show any rows :
In my Model class:
public ObservableCollection<Step2Model> Step2ModelList
{
get {
if (step2ModelList == null)
step2ModelList = new ObservableCollection<Step2Model>();
return step2ModelList;
}
set
{
step2ModelList = value;
Changed("Step2ModelList");
}
}
public void AddStep2ModelToList(Step2Model step2Model)
{
Step2ModelList.Add(step2Model);
}
public void addRowsToList(int count)
{
for (int i = 0; i < count; i++)
{
Step2ModelList.Add(new Step2Model()); // UPDATED
/*
AddItemCommand = new ActionCommand
{
ExecuteDelegate = o => Step2ModelList.Add(new Step2Model())
};
*/
}
}
DataGrid is bound to Step2ModelList
ItemsSource="{Binding Step2ModelList, Mode=TwoWay}">
On combo box selection,
private void cboNumZones_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
int num = (int)cboNumZones.SelectedItem;
Console.WriteLine("Numer of Zones SElected = " + num);
if (num > 0)
{
Step2InfoData st = this.DataContext as Step2InfoData;
st.addRowsToList(count);
}
}
After this the grid doesn't reflect the 2 added rows in Step2ModelList.
I think, you don't understand concept of DataGrid.ItemSource. Read http://msdn.microsoft.com/en-us/library/system.windows.controls.datagrid.itemssource(v=vs.95).aspx
DataGrid.ItemSource is used to populate DataGrid not the other way around.
If you want to access DataGrid items use:
Update, ObservableCollection Sample:
ViewModel:
View code behind:
View:
Full source code may be found at https://github.com/mswietlicki/SimpleMVVMApp