5

D6 教授

以前我们使用 DBISAM 和 DBISAMTable。它处理 RecNo,并且可以很好地进行修改(删除、编辑等)。

现在我们用不处理 RecNo 的 ElevateDB 替换,而且很多时候我们使用查询,而不是表。

查询必须重新打开才能看到修改。

但是如果我们重新打开查询,我们需要重新定位到最后一条记录。仅定位是不够的,因为 Grid 将其显示在另一行中。这是非常令人不安的事情,因为修改记录移动到另一行之后,你很难跟踪它,用户讨厌这个。

我们找到了这段代码:

function TBaseDBGrid.GetActRow: integer;
begin
 Result := -1 + Row;
end;


procedure TBasepDBGrid.SetActRow(aRow: integer);
var
 bm : TBookMark;
begin
 if IsDataSourceValid(DataSource) then with DataSource.DataSet do begin
  bm := GetBookmark;
  DisableControls;
  try
   MoveBy(-aRow);
   MoveBy(aRow);
   //GotoBookmark(bm);
  finally
   FreebookMark(bm);
   EnableControls;
  end;
 end;
end;

原始示例是使用 moveby。这对查询很好,因为我们看不到查询在后台重新打开,视觉控件没有改变行位置。

但是当我们有 EDBTable 或 Live/Sensitive Query 时,使用 MoveBy 很危险,因为如果有人删除或追加新行,我们可能会重新定位到错误的记录中。

然后我尝试使用书签(见备注)。但是这种技术不起作用,因为它是在另一个 Row 位置显示记录......

所以问题是:如何在 DBGrid 中强制行位置和记录?

或者在底层 DataSet 刷新后,什么样的 DBGrid 可以重新定位到记录/行?

我搜索用户友好的解决方案,我理解它们,因为我尝试使用这种跨 DBGrid 的跳转,并且使用起来非常糟糕,因为更新后尝试查找原始记录时我的眼睛都出来了...... :-(

感谢您的每一个帮助,链接,信息:dd

4

4 回答 4

10

由于“MoveBy”对您有用,请使用它们。

在关闭数据集之前获取“书签”。完成您的工作,重新打开数据集,然后使用“MoveBy”将您的记录重新定位在网格上。完成后,获取另一个 Bookmark 并使用 DataSet.CompareBookmarks 将其与前一个进行比较。如果结果是 0 很好,如果不是,那么只有为上一个书签发出“GotoBookmark”。

这样,只要另一个用户没有删除/插入记录,您的网格似乎就不会跳动,如果不是这种情况,至少您会在同一条记录上。


编辑:这是一些代码示例,即使数据集中有删除/插入,也应将所选记录重新定位在正确的位置。请注意,代码省略了禁用/启用控件,以及为简单起见填充网格的记录较少的特殊情况。

type
  TAccessDBGrid = class(TDBGrid);

procedure TForm1.Button1Click(Sender: TObject);
var
  BmSave, Bm: TBookmark;
  GridRow, TotalRow: Integer;
begin
  GridRow := TAccessDBGrid(DBGrid1).Row;
  TotalRow := TAccessDBGrid(DBGrid1).RowCount;
  BmSave := DBGrid1.DataSource.DataSet.GetBookmark;
  try

    // close dataset, open dataset...

    if DBGrid1.DataSource.DataSet.BookmarkValid(BmSave) then
      DBGrid1.DataSource.DataSet.GotoBookmark(BmSave);
    Dec(TotalRow);
    if GridRow < TotalRow div 2 then begin
      DBGrid1.DataSource.DataSet.MoveBy(TotalRow - GridRow);
      DBGrid1.DataSource.DataSet.MoveBy(GridRow - TotalRow);
    end else begin
      if dgTitles in DBGrid1.Options then
        Dec(GridRow);
      DBGrid1.DataSource.DataSet.MoveBy(-GridRow);
      DBGrid1.DataSource.DataSet.MoveBy(GridRow);
    end;
    Bm := DBGrid1.DataSource.DataSet.GetBookmark;
    try
      if (DBGrid1.DataSource.DataSet.BookmarkValid(Bm) and
          DBGrid1.DataSource.DataSet.BookmarkValid(BmSave)) and
          (DBGrid1.DataSource.DataSet.CompareBookmarks(Bm, BmSave) <> 0) then
        DBGrid1.DataSource.DataSet.GotoBookmark(BmSave);
    finally
      DBGrid1.DataSource.DataSet.FreeBookmark(Bm);
    end;
  finally
    DBGrid1.DataSource.DataSet.FreeBookmark(BmSave);
  end;
end;
于 2010-06-08T10:06:05.910 回答
1

在关闭和重新打开查询之前存储您的唯一键字段的值,然后Locate在重新打开后存储到记录中。DisableControls/EnableControls防止屏幕更新。

于 2010-06-08T08:30:57.907 回答
0

我想到的只是一段简单的代码:

procedure DoRefresh(Dataset: TDataset);
var
  bkm: TBookmark;
begin
  Dataset.UpdateCursorPos;
  bkm := Dataset.GetBookmark;
  Dataset.DisableControls;
  try
    Dataset.Refresh;  //refresh dataset if it's open

    if Dataset.BookmarkValid(bkm) then
    begin
      Dataset.GotoBookmark(bkm);
    end;
  finally
    Dataset.EnableControls;
    Dataset.FreeBookmark(bkm);
  end;
end;
于 2010-06-08T09:00:54.650 回答
0

记录位置很大程度上取决于您从 Query/Table 对象获得的结果集的排序顺序。如果您根本不订购,那么您从服务器获得的顺序是实现定义的,因此,即使没有发生任何更改,也不能保证重新打开查询时记录的顺序相同。至少在 MSSQL 和 Firebird 中,如果没有使用Order By子句,结果会以不同的顺序出现。

As for repositioning, I think that TOndrej solution is the safest one - using the primary key of your resultset to reposition the grid on the right record.

于 2010-06-08T18:39:14.897 回答