DataGridView in TabControl and CellValidating lead to problems
I created a little form with a TabControl on it and a combobox.
On the first page i added a DataGridView with 2 columns, not bound to any data (data entered directly).
Also, for the grid i added an event handler for CellValidating in which i test if the data from the first row is numeric, if not i show a message box with an error and return e.Cancel = true so the DataGridView maintains focus.
Now, if i do the following :
1 - edit cell on first column and enter a wrong value (a text)
2 - click on TabControl to change to the other tab page, error is shown that the value is not correct, close the error
3 - modify the value to be correct (enter a number)
4 - change to any of the other cells in the DataGridView and try to edit them with the mouse (double click to edit), it just doesn't work.
The only solution to "fix" this is to change to the other tabPage on the TabControl and back, this way the DataGridView regains its ability to edit cells with the mouse.
Is there any solution to this ?
Note: If i don't show the message box in the CellValidating function then the problem doesn't occur. I know i could show an errormessage on the row of the column, but the application i'm writing requires that i display messageboxes for all errors (and i can't make the grid work differently from all other controls on the other dialogs just because of this bug).
So, is there any way to fix this ?
Here's my code (at least the part that relates to the problem)
public Form1()
{
InitializeComponent(); // add some data to the grid so it's not empty // first row
DataGridViewRow row = new DataGridViewRow();
DataGridViewTextBoxCell value = new DataGridViewTextBoxCell();
DataGridViewTextBoxCell text = new DataGridViewTextBoxCell();
value.Value = "12";
text.Value = "some text"; row.Cells.Add(value);
row.Cells.Add(text); dataGridView1.Rows.Add(row); // second row
row = new DataGridViewRow();
value = new DataGridViewTextBoxCell();
text = new DataGridViewTextBoxCell();
value.Value = "1000";
text.Value = "another text"; row.Cells.Add(value);
row.Cells.Add(text); dataGridView1.Rows.Add(row);
} private void dataGridView1_CellValidating(object sender, DataGridViewCellValidatingEventArgs e)
{
// first row must be numeric value
if (e.ColumnIndex == 0)
{
try
{
double x = Convert.ToDouble(e.FormattedValue);
}
catch (System.Exception ex)
{
MessageBox.Show("Error, value is not a double");
// if we can't convert then bail out
e.Cancel = true;
}
}
} ...
private void InitializeComponent()
{
...
this.dataGridView1.CellValidating += new System.Windows.Forms.DataGridViewCellValidatingEventHandler(this.dataGridView1_CellValidating);
...
}
该问题源自MSDN Bug
http://social.msdn.microsoft.com/Forums/windows/en-US/c3634471-49cb-4274-afa4-c4bcd184b52c/datagridview-in-tabcontrol-and-cellvalidating-lead-to-problems
如果觉得英文不爽,这有这个问题的中文版
http://bbs.csdn.net/topics/390492161?page=1#post-394804656
下面是我对这个问题的规避方法
首先确定问题是在切换页签引发的校验中,弹出弹窗,导致焦点失去,从而致使datagridview Cell无法再进入编辑模式。
所以可以通过在校验过程弹窗后,重新fouce。
private void dataGridView1_CellValidating(object sender, DataGridViewCellValidatingEventArgs e)
{if(e.ColumnIndex==-||e.RowIndex==-)
{
return;
}
if(string.IsNullOrEmpty(e.FormattedValue.ToString()))
{
return;
}
int tempValue = -;
if(!int.TryParse(e.FormattedValue.ToString(),out tempValue))
{
MessageBox.Show("Erro Data.");
dataGridView1.Focus();
e.Cancel = true;
dataGridView1.CancelEdit();
}
}
但是fouce后会导致tab页签发生切换,而不是我们想要的留在该页签(这里是因为焦点重获得,导致某个标志位值被修改,tabControl未进入留在本页签的代码分支),所以需要为tabControl写Deselecting事件来“要求停留在本页签”,这里需要一个标志位,来实现如果校验通过,那么页签正常切换,如果失败不能切换。
private void dataGridView1_CellValidating(object sender, DataGridViewCellValidatingEventArgs e)
{
m_IsInvalid = false;
if(e.ColumnIndex==-||e.RowIndex==-)
{
return;
}
if(string.IsNullOrEmpty(e.FormattedValue.ToString()))
{
return;
}
int tempValue = -;
if(!int.TryParse(e.FormattedValue.ToString(),out tempValue))
{
MessageBox.Show("Erro Data.");
dataGridView1.Focus();
e.Cancel = true;
m_IsInvalid = true;
dataGridView1.CancelEdit();
}
}
private void tabControl1_Deselecting(object sender, TabControlCancelEventArgs e)
{
if (e.TabPage == tabPage2)
{
if (m_IsInvalid)
{
e.Cancel = true;
}
}
}
private void dataGridView1_RowValidating(object sender, DataGridViewCellCancelEventArgs e)
{
m_IsInvalid = false;
if (e.ColumnIndex == - || e.RowIndex == -)
{
return;
}
MessageBox.Show("Erro Data.");
dataGridView1.Focus();
e.Cancel = true;
m_IsInvalid = true;
dataGridView1.CancelEdit();
}
这样就规避了页签切换校验失败,导致无法点击鼠标编辑datagridview cell的问题。
当然,坑一般是连着的,datagridview的CausesValidation一旦设为true,那么就是一旦焦点失去,它就触发校验,这样用户体验感非常差
想想看,我表格填错了,想直接点cancel取消掉,都要各种弹窗,无法正常Cancel。所以我们一般做了这样的策略
就是如果是在表格内进行操作,那么焦点失去立即校验。如果是在表格外,那么取消这一坑爹设定。
private void dataGridView1_MouseLeave(object sender, EventArgs e)
{
dataGridView1.CausesValidation = false;
} private void dataGridView1_MouseEnter(object sender, EventArgs e)
{
dataGridView1.CausesValidation = true;
}
好,这样,我们的Cancel,关闭窗体的X,就能正常运作了。
但是,但是。。。。我们发现页签切换点击,也是在表格外,这样就会导致页签切换时,触发一个行校验后,再点击页签切换,就不会触发表格校验,同时因为我们之前设置的全局变量导致页签一直无法切换,也没有提示。
我们必须在点击页签切换时,将dataGridView1.CausesValidation从false重新赋值为true。很悲剧的是,这样的事件微软没有给出。所以必须要重写下tabControl控件
public class CustomTabControl:TabControl
{
public event EventHandler<CancelEventArgs> PreClicked; protected override void WndProc(ref Message m)
{
if (m.Msg == )
{
var cancelArg = new CancelEventArgs(false);
OnPreClicked(cancelArg);
if (cancelArg.Cancel)
{
return;
}
}
base.WndProc(ref m);
}
private void OnPreClicked(CancelEventArgs e)
{
var handler = PreClicked;
if(handler!=null)
{
handler(this,e );
}
}
}
Form1窗体:
private void tabControl1_PreClicked(object sender, CancelEventArgs e)
{
dataGridView1.CausesValidation = true;
}
这样所有就能按照设计正常运行了。
DataGridView in TabControl and CellValidating lead to problems的更多相关文章
- The `XXXX` target overrides the `HEADER_SEARCH_PATHS` build setting defined in `Pods/Target Support Files/Pods-game-desktop/Pods-game-desktop.release.xcconfig'. This can lead to prob
The `game-desktop [Release]` target overrides the `HEADER_SEARCH_PATHS` build setting defined in `Po ...
- DataGridView使用
DataGridView控件概述 DataGridView 控件代码目录(Windows 窗体) 未绑定数据列 定义:可能想要显示并非来自数据源的一列数据,这种列称为未绑定列. 数据格式示例 如何:设 ...
- Android Lint Checks
Android Lint Checks Here are the current list of checks that lint performs as of Android Studio 2.3 ...
- iOS cocoapods升级及问题
安装 安装RubyCocoaPods基于Ruby语言开发而成,因此安装CocoaPods前需要安装Ruby环境.幸运的是Mac系统默认自带Ruby环境,如果没有请自行查找安装.检测是否安装Ruby:$ ...
- CocoaPods的使用及安装
本文转自:http://www.jianshu.com/p/6e5c0f78200a 一.什么是CocoaPods CocoaPods是iOS项目的依赖管理工具,该项目源码在Github上管理.开发i ...
- android 官方文档 JNI TIPS
文章地址 http://developer.android.com/training/articles/perf-jni.html JNI Tips JNI is the Java Native I ...
- iOS 如何在一个已经存在多个project的workspace中引入cocoapods管理第三方类库
一种新的第三方库管理工具:Carthage 如何使用Carthage管理iOS依赖库 Podfile Syntax Reference v1.1.0.rc.3 https://guides.cocoa ...
- cocoapod集成失败,无法找到头文件的解决办法
在终端更新pod的时候,提示警告: target overrides the `OTHER_LDFLAGS` build setting defined in `Pods/Target Support ...
- ubuntu-利用pdnsd-TCP方式获取IP-拒绝DNS污染
那,自从国内技术出现了DNS污染问题呢,时常导致很多国外网站访问不正常,所以通过参考一些博客所属避免DNS污染的方法,决定搭建一个Ubuntu JeOS下的DNS缓存服务器,该服务器利用TCP方式获取 ...
随机推荐
- WLAN频段的选择
WLAN(无线局域网)通过射频技术,将设备之间互联.当前应用于WLAN组网的有WIFI,蓝牙等.蓝牙工作在2.4GHZ的ISM(工业.科研.医疗)频段,功率等级分为0dBm,10dBm,20dBm:发 ...
- LCC
LCC: super vector:
- [JSP]Maven+SSM框架(Spring+SpringMVC+MyBatis) - Hello World
来源:http://blog.csdn.net/zhshulin/article/details/37956105?utm_source=tuicool&utm_medium=referral ...
- Python算法-冒泡排序
#coding:utf-8 """ 冒泡排序 原理:依次重复访问每一个需要排序的元素,每次比较相邻的两个元素是否符合顺序,若不符合就交换,直到没有不符合顺序的为止. &q ...
- vim的树形菜单NERDTREE的设置
网上比较好的一篇文章:http://coolshell.cn/articles/1679.html http://coolshell.cn/articles/11312.html 1.Vim安装NER ...
- mybatis执行批量更新update
Mybatis的批量插入这里有http://ljhzzyx.blog.163.com/blog/static/38380312201353536375/.目前想批量更新,如果update的值是相同的话 ...
- Java线程问题分析定位
Java线程问题分析定位 分析步骤: 1.使用top命令查看系统资源占用情况,发现Java进程占用大量CPU资源,PID为11572: 2.显示进程详细列表命令:ps -mp 11572 -o THR ...
- dock停靠管理器
DockManager停靠管理器可以对它所拥有的 停靠面板 的行为和外观设置进行集中控制.DockPanel停靠面板是停靠应用程序的主要构成部件. 常规面板 DockPanel.ParentPanel ...
- {HDU}{2516}{取石子游戏}{斐波那契博弈}
题意:给定一堆石子,每个人最多取前一个人取石子数的2被,最少取一个,最后取石子的为赢家,求赢家. 思路:斐波那契博弈,这个题的证明过程太精彩了! 一个重要的定理:任何正整数都可以表示为若干个不连续的斐 ...
- IIS7.5 webapi 不支持 Delete、Put 解决方法
在IIS管理界面选择API的项目,选择 “Features View”. 2. 选择 “Handler Mappings” 菜单. 3. 打开“WebDAV” 选项. 4. 点击 “Request ...