c# wpf datagrid 模板列修改某个单元格,更新所选行另一个单元格的值,如何做到呢?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了c# wpf datagrid 模板列修改某个单元格,更新所选行另一个单元格的值,如何做到呢?相关的知识,希望对你有一定的参考价值。

我们知道datagrid是通过绑定后台数据集合来实现前台显示的,而每一行的datacontext对应的是个体数据。例如一个Student类,它有id和Name属性。datagrid绑定的是Student集合,每行对应某一个具体的student。你的需求是模板列里修改单元格的值,更新另一个单元的值。还是拿student举例。例如我们的name要求根据id+1,而id值是可以在前台修改的值。它放在模板列里的textbox供我们去修改。(前台修改导致后台数据更新的话就涉及到双向绑定,不懂的话自己先去网上查查相关资料)那我们就可以通过Textbox的TextChanged事件对这行所对应的student的name做操作。或者通过id的value变化直接在id的Set函数里做操作。 参考技术A //如果你的模板是这样布局
<ListView>
    <ListView.ItemTemplate>
        <DateTemplete>
            <Grid>
                <Grid.ColumnDefinitions>
                    <ColumDefinition/>
                    <ColumDefinition/>
                    <ColumDefinition/>
                </Grid.ColumnDefinitions>
                <TextBlock Grid.Colum="0" Text=Binding Id/>
                <TextBox Grid.Colum="1" TextChanged="TextBox_TextChanged_1"/>
                <TextBox Grid.Colum="2"/>
            </Grid>
        </DataTemplete>
    </ListView.ItemTemplate>
</ListView>


private void TextBox_TextChanged_1(object sender, TextChangedEventArgs e)

    var txtBox = (TextBox)sender;
    //找到父容器
    var grid = (Grid)txtBox.Parent;
    //根据父容器获取控件
    var txtBox2 = (TextBox)grid.Children[2];
    txtBox2.Text = "My Code";

参考技术B SelectedItem,获取选中行。

C# WPF DataGrid获取单元格并改变背景色

01

概述

WPF 自带了一个表格控件datagrid,这个控件类似winfrom中的datagridview,在数据显示的时候也经常会用到,这节主要讲解如何从后台代码获取到单元格控件并改变其相关属性:背景色、前景色、字体等。

02


效果演示

03


代码

后台cs:

using System;
using System.Linq;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Controls.Primitives;
using System.Windows.Media;


namespace Caliburn.Micro.Hello

    /// <summary>
    /// ShellView.xaml 的交互逻辑
    /// </summary>
    public partial class ShellView : UserControl
    
        public ShellView()
        
            InitializeComponent();
        


        private void DG_SelectedCellsChanged(object sender, SelectedCellsChangedEventArgs e)
        
            int colindex = -1;
            int rowindex = -1;


            //方法1
            //DataGridCellInfo info = new DataGridCellInfo(dg.Items[0], dg.Columns[2]);
            //方法2
            //foreach (DataGridCellInfo info in this.dgSourceData.SelectedCells)
            //
            //    string str = ((TextBlock)info.Column.GetCellContent(info.Item)).Text;
            //    Console.WriteLine(str);
            //
            //方案1
            var info = this.dgSourceData.SelectedCells.FirstOrDefault();
            var str = ((TextBlock)info.Column.GetCellContent(info.Item)).Text;
            //((TextBlock)info.Column.GetCellContent(info.Item)).Foreground = new SolidColorBrush(Colors.Red);
            //Console.WriteLine(str);


            //方案2
            colindex = this.dgSourceData.CurrentCell.Column.DisplayIndex;//获取选中单元格列号
            //rowindex = this.dgSourceData.SelectedIndex;//获取选中单元格行号
            for (int i = 0; i < ShellViewModel.StudentList.Count(); i++)
            
                if (ShellViewModel.StudentList[i] == this.dgSourceData.CurrentItem)
                
                    //MessageBox.Show("当前选择的行是:" + i.ToString());
                    rowindex = i;
                
            
            DataGridRow row = (DataGridRow)dgSourceData.ItemContainerGenerator.ContainerFromIndex(rowindex);//获取选中单元格所在行
            DataGridCellsPresenter presenter = GetVisualChild<DataGridCellsPresenter>(row);//函数调用,获取行中所有单元格的集合
            DataGridCell cell = (DataGridCell)presenter.ItemContainerGenerator.ContainerFromIndex(colindex);//锁定选中单元格(重点)
            if (cell != null)
            
                TextBlock tb = cell.Content as TextBlock;
                Console.WriteLine(tb.Text);
                dgSourceData.ScrollIntoView(row, dgSourceData.Columns[colindex]);
                //cell = (DataGridCell)presenter.ItemContainerGenerator.ContainerFromIndex(colindex);
                cell.Focus();
                cell.Background = new SolidColorBrush(Colors.Red);//OK!问题解决,选中单元格变色
                cell.Foreground = new SolidColorBrush(Colors.Yellow);
                cell.FontSize = 20;
            
        
        private void dgSourceData_BeginningEdit(object sender, DataGridCellEditEndingEventArgs e)
        
           


        
        /// <summary>
        /// 获取父可视对象中第一个指定类型的子可视对象
        /// </summary>
        /// <typeparam name="T">可视对象类型</typeparam>
        /// <param name="parent">父可视对象</param>
        /// <returns>第一个指定类型的子可视对象</returns>
        public static T GetVisualChild<T>(Visual parent) where T : Visual
        
            T childContent = default(T);
            int numVisuals = VisualTreeHelper.GetChildrenCount(parent);
            for (int i = 0; i < numVisuals; i++)
            
                Visual v = (Visual)VisualTreeHelper.GetChild(parent, i);
                childContent = v as T;
                if (childContent == null)
                
                    childContent = GetVisualChild<T>(v);
                
                if (childContent != null)
                 break; 
            
            return childContent;
        


        public void dgSourceData_SelectionChanged(object sender, SelectionChangedEventArgs e)
        
            MessageBox.Show(1111.ToString());//SelectionMode="Extended" SelectionUnit="Cell" 模式下触发不了
        
    

前台xaml:

<DataGrid Name="dgSourceData" AutoGenerateColumns="False" ItemsSource="Binding StudentList,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged" 
                  ContextMenu="Binding menu1" RowHeaderWidth="30"  SelectedItem ="Binding SelectedItems" SelectionMode="Single" SelectionUnit="Cell"
                                cal:Message.Attach="[Event SelectionChanged]=[GridControl_SelectionChanged($source,$eventArgs)];" 
                               SelectedCellsChanged="DG_SelectedCellsChanged" CellEditEnding="dgSourceData_BeginningEdit" SelectionChanged="dgSourceData_SelectionChanged" >
            <DataGrid.Columns>
                <DataGridTextColumn Header="Name"  Binding=" Binding Path=Name,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged" MinWidth="68"/>
                <DataGridTextColumn Header="Age" Binding=" Binding Path=Age,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged" MinWidth="68"/>
                <DataGridTextColumn Header="Id"  Binding=" Binding Path=Id,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged" MinWidth="48"/>
            </DataGrid.Columns>
        </DataGrid>

04


解析

① 因为我需要绑定SelectedCellsChanged事件,所以前台将默认的行选中模式修改为单元格选中模式:

SelectionMode="Single" SelectionUnit="Cell"

这样修改后SelectionChanged这个事件将不再触发(只有在行选中模式下生效);

②获取选中单元格的值:

var info = this.dgSourceData.SelectedCells.FirstOrDefault();
 var str = ((TextBlock)info.Column.GetCellContent(info.Item)).Text;

③获取选中的列号:

colindex = this.dgSourceData.CurrentCell.Column.DisplayIndex;//获取选中单元格列号

④获取选中行的行号

在SelectionUnit="FullRow" 时候:

可以通过这样获取:

rowindex = this.dgSourceData.SelectedIndex;//获取选中单元格行号

在当SelectionUnit="Cell"时:

我是通过选中单元格对应行的信息和表格控件绑定的集合匹配获取行号的:

for (int i = 0; i < ShellViewModel.StudentList.Count(); i++)
            
                if (ShellViewModel.StudentList[i] == this.dgSourceData.CurrentItem)
                
                    //MessageBox.Show("当前选择的行是:" + i.ToString());
                    rowindex = i;
                
            

⑤获取选中单元格并改变字体颜色:

var info = this.dgSourceData.SelectedCells.FirstOrDefault();
 var str = ((TextBlock)info.Column.GetCellContent(info.Item)).Text;
 ((TextBlock)info.Column.GetCellContent(info.Item)).Foreground = new SolidColorBrush(Colors.Red);

05


源码下载

网盘下载链接:https://pan.baidu.com/s/1TD2BT5hiT-z-_7Z2Bn3PVQ

提取码:添加小编微信获取

技术群:添加小编微信并备注进群

小编微信:mm1552923   

公众号:dotNet编程大全      

以上是关于c# wpf datagrid 模板列修改某个单元格,更新所选行另一个单元格的值,如何做到呢?的主要内容,如果未能解决你的问题,请参考以下文章

WPF的DataGrid列头和行头相交的单元格怎么修改样式和填充文字

WPF DataGrid表头合并且动态添加列

C# WPF datagrid checkboxcolumn 使用问题

C# wpf datagrid 动态加载数据后改变单元格颜色bug

使用 C# 在 WPF 中使 DataGrid 列标题可排序

WPF Datagrid 模板列编辑事件和交替列颜色