Delphi - restore actual row in DBGrid

后端 未结 4 2013
野趣味
野趣味 2021-01-03 03:47

D6 prof.

Formerly we used DBISAM and DBISAMTable. That handle the RecNo, and it is working good with modifications (Delete, edit, etc).

Now we replaced with

相关标签:
4条回答
  • 2021-01-03 03:52

    Just simple piece of code that came in my mind:

    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;
    
    0 讨论(0)
  • 2021-01-03 04:02

    Record position depends much on the sort order of resultset you got from the Query/Table object. If you don't order at all, the order you get from the server is implementation defined and such, can't guarantee that records come in the same order when reopen the query, even if no changes happened. At least in MSSQL and Firebird, results come in different orders if no Order By clause is used.

    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.

    0 讨论(0)
  • 2021-01-03 04:04

    Store the value(s) of your unique key field(s) before closing and reopening the query, then Locate to the record after reopening. DisableControls/EnableControls to prevent screen updates.

    0 讨论(0)
  • 2021-01-03 04:05

    Since 'MoveBy's are working for you, use them.

    Get a 'Bookmark' before closing the dataset. Do your work, reopen the dataset and then reposition your record on the grid with 'MoveBy's. When you're done, get another Bookmark and compare it with the previous one with DataSet.CompareBookmarks. If the result is 0 fine, if not, only then issue a 'GotoBookmark' for the previous bookmark.

    This way, as long as another user have not deleted/inserted records your grid will not seem to be jumpy, and if this is not the case at least you'd be on the same record.


    edit: Here's some code sample that should reposition the selected record in the correct place even when there had been deletes/inserts in the dataset. Note that the code omits disabling/enabling controls, and the special case when there are less records to fill the grid for simplicity.

    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;
    
    0 讨论(0)
提交回复
热议问题