5

我用 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 为 nil。DragControl 是 TControl 类型的全局变量。我尝试使用assigncheck 修补此方法并在DragControl = nil 时调用CancelDrag,但这也失败了,因为DragObject 也是nil。

procedure CancelDrag;
begin
 if DragObject <> nil then DragDone(False);
 DragControl := nil;
end;

为了找出 DragControl 为零的原因,我检查了 DragInitControl。如果 DragControl 为 nil,则有 2 行将退出。

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;

可能是原因......所以我的问题。

  1. 有没有人遇到过类似的拖放问题?
  2. 如果我检测到 DragControl = nil 如何取消当前的拖放?

编辑: 目前我对此没有解决方案,但我可以添加更多关于它的信息。这些网格称为超网格。这是我们为满足我们的需求而开发的内部组件。它从 Devexpress 继承 TcxGrid。我认为(但不确定)当用户在网格重新加载数据的同时拖动网格行时会出现此问题。不知何故,对当前行的引用变成了 nil。从长远来看,我们计划用同样继承自 TcxGrid 的 Bold 感知网格(因为我们在 Delphi 中使用 Bold)替换这个超级网格。然后,一旦数据更改(用户或代码中没有刷新),网格就会更新,并希望这可以解决问题。

4

1 回答 1

3
  1. 不,我在使用 VCL 拖放时从来没有遇到过任何(这些)问题,我对此有相当多的经验。

  2. DragControl是控制单元本地的,那么如何DragControl = nil在生产代码中检测?通常情况下,不需要检查它,至少我从来不需要。ESC通过调用CancelDrag. 正如您已经注意到的那样,该例程DragDone仅在DragObject <> nil. 因此,看似DragObjectnil 已经在说没有正在进行的拖动操作(不再)。

此外,您认为 AV 的来源来自特定行的观察Controls.DragTo似乎是错误的。在正常的拖放操作中,DragControl存在nil不会导致 AV。但是,Controls.DragFindTarget在拖拽和停靠操作中可能会出现问题,但您没有提到进行任何停靠。

您能否澄清一下在什么情况下,或者这个“错误”出现的代码是什么?

于 2011-12-24T11:41:40.937 回答