我应用程序是使用Delphi 2007编译的,其中包含网格之间的拖放功能,大多数时候运行正常。但有时会出现“访问冲突”的错误。我进行了调试,并定位到了VCL中Controls.pas方法DragTo。
代码如下:
begin
if (ActiveDrag <> dopNone) or (Abs(DragStartPos.X - Pos.X) >= DragThreshold) or
(Abs(DragStartPos.Y - Pos.Y) >= DragThreshold) then
begin
Target := DragFindTarget(Pos, TargetHandle, DragControl.DragKind, DragControl);
异常发生在最后一行,因为DragControl为空。DragControl是一个类型为TControl的全局变量。 我已经尝试使用assigncheck来修补此方法,并在DragControl = nil时调用CancelDrag,但这也失败了,因为DragObject也为空。
procedure CancelDrag;
begin
if DragObject <> nil then DragDone(False);
DragControl := nil;
end;
为了找出DragControl为什么是nil,我检查了DragInitControl。如果DragControl是nil,那么有两行代码会直接退出。
procedure DragInitControl(Control: TControl; Immediate: Boolean; Threshold: Integer);
var
DragObject: TDragObject;
StartPos: TPoint;
begin
DragControl := Control;
try
DragObject := nil;
DragInternalObject := False;
if Control.FDragKind = dkDrag then
begin
Control.DoStartDrag(DragObject);
if DragControl = nil then Exit;
if DragObject = nil then
begin
DragObject := TDragControlObjectEx.Create(Control);
DragInternalObject := True;
end
end
else
begin
Control.DoStartDock(DragObject);
if DragControl = nil then Exit;
if DragObject = nil then
begin
DragObject := TDragDockObjectEx.Create(Control);
DragInternalObject := True;
end;
with TDragDockObject(DragObject) do
begin
if Control is TWinControl then
GetWindowRect(TWinControl(Control).Handle, FDockRect)
else
begin
if (Control.Parent = nil) and not (Control is TWinControl) then
begin
GetCursorPos(StartPos);
FDockRect.TopLeft := StartPos;
end
else
FDockRect.TopLeft := Control.ClientToScreen(Point(0, 0));
FDockRect.BottomRight := Point(FDockRect.Left + Control.Width,
FDockRect.Top + Control.Height);
end;
FEraseDockRect := FDockRect;
end;
end;
DragInit(DragObject, Immediate, Threshold);
except
DragControl := nil;
raise;
end;
end;
可能是因为...所以我的问题是。
- 有人遇到过拖放的类似问题吗?
- 如果我检测到DragControl = nil,如何取消当前的拖放操作?
编辑: 目前我没有解决方案,但我可以添加一些关于它的更多信息。这个网格被称为超级网格。这是我们开发的内部组件,以满足我们的需求。它继承了Devexpress的TcxGrid。我认为(但不确定)当用户同时拖动网格行并重新加载数据时,就会出现这个问题。某种方式,对当前行的引用变为nil。长期来看,我们计划用一个Bold感知网格(因为我们使用Delphi的Bold)替换这个超级网格,它也继承自TcxGrid。然后,网格在数据更改时立即更新(用户或代码中没有刷新),希望这样能解决问题。