gpt4 book ai didi

delphi - 带按钮的字符串网格

转载 作者:行者123 更新时间:2023-12-03 15:50:47 25 4
gpt4 key购买 nike

第一个问题:

如何调用stringgrid中不可见的部分?您需要滚动才能看到它。
例如:
字符串网格中有 20 行,但一次只能看到 10 行。您需要滚动才能看到其他 10 个。“隐藏”的如何称呼?

第二个问题:

我知道这可能不是正确的方法,因此我们将不胜感激。
我有一个带有 1 个固定行的字符串网格。我在运行时添加 ColorButtons。所以我用按钮填充 1 列。我使用此按钮来“插入/删除”行。只要所有网格都在“可见”部分,这种方法就可以正常工作。当我“插入”新行并将按钮移动到“隐藏”部分时,出现问题。然后将最后一个按钮绘制到 Cell[0,0]。 “隐藏”部分中的其他按钮绘制正确。知道为什么会发生这种情况吗?我应该在 OnDraw 方法中找到一种方法来管理这个问题,还是有更好(正确)的方法来做到这一点?

代码:

procedure Tform1.addButton(Grid : TStringGrid; ACol : Integer; ARow : Integer);
var
bt : TColorButton;
Rect : TRect;
index : Integer;
begin
Rect := Grid.CellRect(ACol,ARow);
bt := TColorButton.Create(Grid);
bt.Parent := Grid;
bt.BackColor := clCream;
bt.Font.Size := 14;
bt.Width := 50;
bt.Top := Rect.Top;
bt.Left := Rect.Left;
bt.Caption := '+';
bt.Name := 'bt'+IntToStr(ARow);
index := Grid.ComponentCount-1;
bt :=(Grid.Components[index] as TColorButton);
Grid.Objects[ACol,ARow] := Grid.Components[index];
bt.OnMouseUp := Grid.OnMouseUp;
bt.OnMouseMove := Grid.OnMouseMove;
bt.Visible := true;
end;

procedure MoveRowPlus(Grid : TStringGrid; Arow : Integer; stRow : Integer);
var
r, index : Integer;
bt : TColorButton;
Rect : TRect;
begin
Grid.RowCount := Grid.RowCount+stRow;

for r := Grid.RowCount - 1 downto ARow+stRow do
begin
Grid.Rows[r] := Grid.Rows[r-StRow];
end;

index := Grid.ComponentCount-1;
for r := Grid.RowCount - 1 downto ARow+stRow do
begin
bt :=(Grid.Components[index] as TColorButton);
Rect := Grid.CellRect(10,r);
bt.Top := Rect.Top;
bt.Left := Rect.Left;
Grid.Objects[10,r] := Grid.Components[index];
dec(index);
end;
for r := ARow to (ARow +stRow-1) do
begin
Grid.Rows[r].Clear;
end;
end;

procedure MoveRowMinus(Grid : TStringGrid; Arow : Integer; stRow : Integer);
var
r, index : Integer;
bt : TColorButton;
Rect : TRect;
begin

for r := ARow to Grid.RowCount-stRow-1 do
begin
Grid.Rows[r] := Grid.Rows[r+StRow];
end;

index := ARow-1;
for r := ARow to Grid.RowCount-stRow-1 do
begin
Rect := Grid.CellRect(10,r);
bt :=(Grid.Components[index] as TColorButton);
bt.Top := Rect.Top;
bt.Left := Rect.Left;
Grid.Objects[10,r] := Grid.Components[index];
bt.Visible := true;
inc(index);
end;

for r := Grid.RowCount-stRow to Grid.RowCount-1 do
begin
Grid.Rows[r].Clear;
end;
Grid.RowCount := Grid.RowCount-stRow;
end;

最佳答案

  1. 对于可见部分,存在 VisibleRowCountVisibleColCount 属性。 TGridAxisDrawInfo 记录类型将可见部分命名为 Boundary 并将所有部分一起命名为 Extent(反之亦然,我不记得了)。因此,VCL 声明的字符串网格的不可见部分没有特定的名称。它只是不可见的部分

  2. 我认为您犯了一个逻辑错误:当您滚动网格时,按钮不会移动。尽管看起来它们在移动,但这只是由于内部调用 ScrollWindow 而移动设备上下文内容的结果。字符串网格组件中的滚动条是自定义添加的,并且不像其他滚动条那样工作。一个TScrollBox

    要始终在其实际位置显示所有按钮,请在 OnTopLeftChanged 事件中重新绘制字符串网格:

    procedure TForm1.StringGrid1TopLeftChanged(Sender: TObject);
    begin
    StringGrid1.Repaint;
    end;

    当所有行的行高和字符串网格的高度永远不会改变时,那么只需创建所有按钮一次就足够了,并让它们保持在原来的位置。这意味着每个按钮不再“附加”到一行,并且将它们存储在 Objects 属性中不再具有任何意义。按下按钮时,只需根据按钮的位置结合字符串网格的 TopRow 属性计算预期的行索引,该属性指定网格中第一个可见可滚动行的索引。/p>

    如果网格可以调整大小,例如通过 anchor ,然后更新父级的 OnResize 事件中的按钮计数。如果字符串网格的行数可能会小于最大可见行数,则还要更新(可见)按钮数。

    如果您想要更多答案,请更新您的问题并解释如何由于与网格和/或按钮的交互而调用 MoveRowPlusMoveRowMinus 例程,因为现在我还不完全明白你想要什么。

    关于 CellRect 给出错误的坐标:那是因为 CellRect 仅适用于完整(或部分)可见单元格。引用documentation :

    If the indicated cell is not visible, CellRect returns an empty rectangle.

<小时/>

根据OP的评论添加

我认为下面的代码可以满足您的要求。每个按钮的原始行索引存储在 Tag 属性中。

unit Unit1;

interface

uses
Windows, Classes, Controls, Forms, StdCtrls, Grids;

type
TForm1 = class(TForm)
Grid: TStringGrid;
procedure GridTopLeftChanged(Sender: TObject);
procedure FormCreate(Sender: TObject);
private
FPrevTopRow: Integer;
procedure CreateGridButtons(ACol: Integer);
procedure GridButtonClick(Sender: TObject);
procedure RearrangeGridButtons;
function GetInsertRowCount(ARow: Integer): Integer;
function GridButtonToRow(AButton: TButton): Integer;
procedure MoveGridButtons(ButtonIndex, ARowCount: Integer);
end;

implementation

{$R *.dfm}

type
TStringGridAccess = class(TStringGrid);

procedure TForm1.FormCreate(Sender: TObject);
begin
FPrevTopRow := Grid.TopRow;
CreateGridButtons(2);
end;

procedure TForm1.CreateGridButtons(ACol: Integer);
var
R: TRect;
I: Integer;
Button: TButton;
begin
R := Grid.CellRect(ACol, Grid.FixedRows);
Inc(R.Right, Grid.GridLineWidth);
Inc(R.Bottom, Grid.GridLineWidth);
for I := Grid.FixedRows to Grid.RowCount - 1 do
begin
Button := TButton.Create(Grid);
Button.BoundsRect := R;
Button.Caption := '+';
Button.Tag := I;
Button.ControlStyle := [csClickEvents];
Button.OnClick := GridButtonClick;
Button.Parent := Grid;
Grid.Objects[0, I] := Button;
OffsetRect(R, 0, Grid.DefaultRowHeight + Grid.GridLineWidth);
end;
end;

procedure TForm1.GridButtonClick(Sender: TObject);
var
Button: TButton absolute Sender;
N: Integer;
I: Integer;
begin
N := GetInsertRowCount(Button.Tag);
if Button.Caption = '+' then
begin
Button.Caption := '-';
Grid.RowCount := Grid.RowCount + N;
for I := 1 to N do
TStringGridAccess(Grid).MoveRow(Grid.RowCount - 1,
GridButtonToRow(Button) + 1);
MoveGridButtons(Button.Tag, N);
end
else
begin
Button.Caption := '+';
for I := 1 to N do
TStringGridAccess(Grid).MoveRow(GridButtonToRow(Button) + 1,
Grid.RowCount - 1);
Grid.RowCount := Grid.RowCount - N;
MoveGridButtons(Button.Tag, -N);
end;
end;

procedure TForm1.GridTopLeftChanged(Sender: TObject);
begin
RearrangeGridButtons;
FPrevTopRow := Grid.TopRow;
end;

procedure TForm1.RearrangeGridButtons;
var
I: Integer;
Shift: Integer;
begin
Shift := (Grid.TopRow - FPrevTopRow) *
(Grid.DefaultRowHeight + Grid.GridLineWidth);
for I := 0 to Grid.ControlCount - 1 do
begin
Grid.Controls[I].Top := Grid.Controls[I].Top - Shift;
Grid.Controls[I].Visible := Grid.Controls[I].Top > 0;
end;
end;

function TForm1.GetInsertRowCount(ARow: Integer): Integer;
begin
//This function should return the number of rows which is to be inserted
//below ARow. Note that ARow refers to the original row index, that is:
//without account for already inserted rows. For now, assume three rows:
Result := 3;
end;

function TForm1.GridButtonToRow(AButton: TButton): Integer;
begin
for Result := 0 to Grid.RowCount - 1 do
if Grid.Objects[0, Result] = AButton then
Exit;
Result := -1;
end;

procedure TForm1.MoveGridButtons(ButtonIndex, ARowCount: Integer);
var
I: Integer;
begin
for I := 0 to Grid.ControlCount - 1 do
if Grid.Controls[I].Tag > ButtonIndex then
Grid.Controls[I].Top := Grid.Controls[I].Top +
ARowCount * (Grid.DefaultRowHeight + Grid.GridLineWidth);
end;

end.

但是我可以说,在不使用按钮控件的情况下这也是可能的:我建议在字符串网格的 OnDrawCell 事件中绘制假按钮控件。

关于delphi - 带按钮的字符串网格,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9046861/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com