做一个自定义用户组件时考虑在TreeView组件和DataGridView组件之间支持拖放,DataGridView作为拖放的目标对象。本来想在网络上找现成的代码片段,但遗憾的是找到的几乎全是从DataGridView中拖放数据到其它组件的代码,唉~~,看来想偷懒也不容易,只好自己研究咯。还好不复杂,很快就找到了方法,好了,无需多说,把代码片段贴出来让大家分享。
作为源对象的TreeView组件支持节点拖放的事件ItemDrag代码:
private void tvMetaData_ItemDrag(object sender, ItemDragEventArgs e)
{
if (e.Button.Equals(MouseButtons.Left) && ((TreeNode)e.Item).Nodes.Count == 0 )
{
tvMetaData.DoDragDrop(e.Item, DragDropEffects.Move);
}
}
呵呵,十分简单,判断鼠标左键是否按下,且只允许最底层的节点支持拖放,然后设置DragDropEffects。
目标对象DataGridView的DragEnter事件处理:
private void dtColumns_DragEnter(object sender, DragEventArgs e)
{
//只接受树节点的拖放
if (e.Data.GetDataPresent(typeof(TreeNode)))
{
e.Effect = DragDropEffects.Move;
}
}
无需多说,通过DragEnter来确定是否可接受的拖放操作;
接下来是主要的数据接收程序片段了:
private void dtQueryCondition_DragDrop(object sender, DragEventArgs e)
{
//将屏幕坐标转换为控件坐标之后获取该坐标下DataGridView的行和列
Point ClientPt = dtQueryCondition.PointToClient(new Point(e.X,e.Y));
System.Windows.Forms.DataGridView.HitTestInfo ht = dtQueryCondition.HitTest(ClientPt.X, ClientPt.Y);
if(ht.ColumnIndex.Equals(0))
{
SqlColumn sc = ((SqlColumn)(((TreeNode)e.Data.GetData(typeof(TreeNode))).Tag));
if (bsFilteCondition.Count == 0 || ht.RowIndex > (bsFilteCondition.Count - 1))
{
QueryCondition qc = new QueryCondition();
qc.FieldName = sc.TableName + "." + sc.Name;
bsFilteCondition.Add(qc);
}
else
{
dtQueryCondition.CurrentCell = dtQueryCondition.Rows[ht.RowIndex].Cells[ht.ColumnIndex];
dtQueryCondition.BeginEdit(false);
dtQueryCondition.CurrentCell.Value = sc.TableName + "." + sc.Name;
dtQueryCondition.EndEdit();
}
}
}
函数说明:如果是添加新行,则通过对DataGridView的数据源(我用的是BindingSource)添加一条新记录的方式来处理,如果是修改已有的行,是通过设置CurrentCell、BeginEdit和EndEdit,首先将CurrentCell设置为鼠标拖放时确定的单元格,接下来调用BeginEdit和EndEdit使DataGridView接受拖放操作所设置的单元格数据。
嗯~~~~,很简单吧