2014-01-08 64 views
3

How do I place a Popup in my DataGridTemplateColumn.CellEditingTemplate correctly?一样,我试图在单元格被编辑时在数据网格的单元格下面出现PopUp,当单元格不再被编辑时消失。最后一点是PopUp的内容根据列是动态的,列是通过绑定动态创建的。编辑WPF DataGrid单元格时显示弹出窗口

我开始使用以下XAML,但是我得到一个XamlParseException“将值添加到类型'System.Windows.Controls.ItemCollection'的集合中引发了一个异常”。

<DataGrid ItemsSource="{Binding Path=Options}"> 
    <DataGridTemplateColumn> 
     <DataGridTemplateColumn.CellEditingTemplate> 
      <DataTemplate> 
       <Grid> 
        <Popup Placement="Bottom" IsOpen="True" Width="200" Height="100"> 
         <TextBlock>Somethingn here</TextBlock> 
        </Popup> 
       </Grid> 
      </DataTemplate> 
     </DataGridTemplateColumn.CellEditingTemplate> 
    </DataGridTemplateColumn> 
</DataGrid> 
+0

最简单的办法是设置DataGridCell Focusable属性设置为false ,并且Popup获得焦点。参考这个答案http://stackoverflow.com/a/18027670/1437877 – Abbas

回答

3

我已经想通了,至今运作良好的解决方案。它涉及一些事件处理程序,但代码不必访问视图模型,因此希望它不会引起MVVM纯粹主义者的愤怒。

XAML:

<Grid> 
    <DataGrid ItemsSource="{Binding Path=Options}" BeginningEdit="DataGrid_BeginningEdit" CellEditEnding="DataGrid_CellEditEnding" /> 
    <Popup Name="pop1"> 
     <Border Width="300" Height="200" Background="LemonChiffon" BorderThickness="2" BorderBrush="Black" /> 
    </Popup> 
</Grid> 

MainWindow.xaml.cs事件处理程序:

private void DataGrid_BeginningEdit (Object sender, DataGridBeginningEditEventArgs e) 
{ 
    DataGrid grid = (DataGrid) sender; 
    Popup pop1 = (Popup) grid.FindName("pop1"); 
    pop1.PlacementTarget = grid.GetCell(e.Row.GetIndex(), e.Column.DisplayIndex); 
    pop1.IsOpen = true; 
} 

private void DataGrid_CellEditEnding (Object sender, DataGridCellEditEndingEventArgs e) 
{ 
    Popup pop1 = (Popup) ((DataGrid) sender).FindName("pop1"); 
    pop1.IsOpen = false; 
} 

DataGridExtensions.cs:

/// <summary> 
/// Extension methods for DataGrid 
/// These methods are thanks to http://blogs.msdn.com/b/vinsibal/archive/2008/11/05/wpf-datagrid-new-item-template-sample.aspx 
/// </summary> 
public static class DataGridExtensions 
{ 
    /// <summary> 
    /// Returns a DataGridCell for the given row and column 
    /// </summary> 
    /// <param name="grid">The DataGrid</param> 
    /// <param name="row">The zero-based row index</param> 
    /// <param name="column">The zero-based column index</param> 
    /// <returns>The requested DataGridCell, or null if the indices are out of range</returns> 
    public static DataGridCell GetCell (this DataGrid grid, Int32 row, Int32 column) 
    { 
     DataGridRow gridrow = grid.GetRow(row); 
     if (gridrow != null) 
     { 
      DataGridCellsPresenter presenter = GetVisualChild<DataGridCellsPresenter>(gridrow); 

      // try to get the cell but it may possibly be virtualized 
      DataGridCell cell = (DataGridCell) presenter.ItemContainerGenerator.ContainerFromIndex(column); 
      if (cell == null) 
      { 
       // now try to bring into view and retreive the cell 
       grid.ScrollIntoView(gridrow, grid.Columns[column]); 

       cell = (DataGridCell) presenter.ItemContainerGenerator.ContainerFromIndex(column); 
      } 

      return (cell); 
     } 

     return (null); 
    } 

    /// <summary> 
    /// Gets the DataGridRow based on the given index 
    /// </summary> 
    /// <param name="idx">The zero-based index of the container to get</param> 
    public static DataGridRow GetRow (this DataGrid dataGrid, Int32 idx) 
    { 
     DataGridRow row = (DataGridRow) dataGrid.ItemContainerGenerator.ContainerFromIndex(idx); 
     if (row == null) 
     { 
      // may be virtualized, bring into view and try again 
      dataGrid.ScrollIntoView(dataGrid.Items[idx]); 
      dataGrid.UpdateLayout(); 

      row = (DataGridRow) dataGrid.ItemContainerGenerator.ContainerFromIndex(idx); 
     } 

     return (row); 
    } 

    private static T GetVisualChild<T> (Visual parent) where T : Visual 
    { 
     T child = default(T); 

     Int32 numvisuals = VisualTreeHelper.GetChildrenCount(parent); 
     for (Int32 i = 0; i < numvisuals; ++i) 
     { 
      Visual v = (Visual) VisualTreeHelper.GetChild(parent, i); 
      child = v as T; 
      if (child == null) 
       child = GetVisualChild<T>(v); 
      else 
       break; 
     } 

     return child; 
    } 
} 
0

我相信一个好的方法是将它设置这样

<Grid> 
    <DataGrid Name="DataGridList" ItemsSource="{Binding SettingsList, Mode=TwoWay}" AutoGenerateColumns="False" Height="146"> 
     <!--Resources--> 
     <DataGrid.Resources>    
      <!--DataTemplate--> 
      <DataTemplate x:Key="DateTemplate" > 
       <StackPanel> 
        <TextBlock Text="{Binding YourBinding}"></TextBlock> 
       </StackPanel> 
      </DataTemplate>    
      <!--EditingDateTemplate--> 
      <DataTemplate x:Key="EditingDateTemplate"> 
       <Grid> 
        <Popup Placement="Bottom" IsOpen="True" Width="200" Height="100"> 
         <TextBlock>Something here</TextBlock> 
        </Popup> 
       </Grid> 
      </DataTemplate> 
     </DataGrid.Resources> 
     <!--Datagrid--> 
     <DataGrid.Columns> 
      <DataGridTemplateColumn Header="header 1" CellTemplate="{StaticResource DateTemplate}" CellEditingTemplate="{StaticResource EditingDateTemplate}" /> 
     </DataGrid.Columns> 
    </DataGrid> 
</Grid> 
+0

这似乎做了一些奇怪的行添加行,并没有创建我的动态列;我也不想用弹出窗口替换正常的单元格编辑行为,但是弹出显示补充信息。 –