6

简短版:有没有办法单独控制或修改 LisBox 项目?例如,将它们的Visible 属性分别设置为 False。我在搜索时在Fire Monkey中找到了一个TListBoxItem类,但我不想在 VCL 中使用 Fire Monkey 并想要它。

详细版本:我尝试使用两个 TStringList 和一个 Edit 过滤我的 ListBox,一个 StringList 是全局的以保留原始列表(list_files_global),另一个 StringList 帮助过滤过程(list_files_filter),我的主要文件列表是我的 ListBox(list_files)。onCreate当程序开始存储我的原始列表时,我在事件上创建了我的全局 StringList :

procedure Tfrm_main.FormCreate(Sender: TObject);
Begin
  list_files_global := TStringList.Create;
  list_files_global.Assign(list_files.Items);
End;

并使用 Edit 的onChange事件进行过滤:

procedure Tfrm_main.edit_files_filterChange(Sender: TObject);
Var
  list_files_filter: TStringList;
  i: Integer;
Begin
  list_files_filter := TStringList.Create;
  list_files_filter.Assign(list_files.Items);

  list_files.Clear;

  for i := 0 to list_files_filter.Count - 1 do 
    if pos(edit_files_filter.text, list_files_filter[i]) > 0 then 
      list_files.Items.Add(list_files_filter[i]);

End;

为了关闭过滤器,只需从我最初创建的全局列表中恢复列表:

list_files.Items := list_files_global;

到目前为止,一切正常,但问题是当我试图从过滤列表中编辑/重命名/删除项目时,例如我更改了一个项目:

list_files.Items[i] := '-- Changed Item --';

列表将被编辑,但是当我关闭过滤器时,原始列表将返回并且所有更改都将丢失。所以我想知道有什么合适的方法来解决这个问题吗?像单独隐藏项目或更改项目可见性等...所以我可以更改过滤算法并摆脱所有这些制作额外列表的操作。我在互联网上搜索并查看了一整天的 Delphi 帮助文件,没有任何有用的东西出现。

4

2 回答 2

8

这是我经常做的事情,但使用列表视图而不是列表框。不过基本原理是一样的。

我倾向于将单个项目存储为对象,它们是 Delphi 中的引用类型。我将它们全部保存在一个未过滤的主列表中,该列表拥有对象,而我维护一个过滤列表(不拥有对象)用于显示目的。像@Sertac 一样,我将它与虚拟列表视图结合起来。

要查看它在实践中是如何工作的,请创建一个新的 VCL 应用程序并在主窗体上放置一个列表视图 ( lvDisplay) 和一个编辑控件 ( ):eFilter

带有编辑和列表视图控件的主窗体的屏幕截图

请注意,我在列表视图控件中添加了三列:“姓名”、“年龄”和“颜色”。我也将其设为虚拟 ( OwnerData = True)。

现在为各个数据项定义类:

type
  TDogInfo = class
    Name: string;
    Age: Integer;
    Color: string;
    constructor Create(const AName: string; AAge: Integer; const AColor: string);
    function Matches(const AText: string): Boolean;
  end;

在哪里

{ TDogInfo }

constructor TDogInfo.Create(const AName: string; AAge: Integer;
  const AColor: string);
begin
  Name := AName;
  Age := AAge;
  Color := AColor;
end;

function TDogInfo.Matches(const AText: string): Boolean;
begin
  Result := ContainsText(Name, AText) or ContainsText(Age.ToString, AText) or
    ContainsText(Color, AText);
end;

让我们创建未过滤的狗列表:

TForm1 = class(TForm)
  eFilter: TEdit;
  lvDisplay: TListView;
  procedure FormCreate(Sender: TObject);
  procedure FormDestroy(Sender: TObject);
private
  FList, FFilteredList: TObjectList<TDogInfo>;
public
end;

在哪里

function GetRandomDogName: string;
const
  DogNames: array[0..5] of string = ('Buster', 'Fido', 'Pluto', 'Spot', 'Bill', 'Rover');
begin
  Result := DogNames[Random(Length(DogNames))];
end;

function GetRandomDogColor: string;
const
  DogColors: array[0..2] of string = ('Brown', 'Grey', 'Black');
begin
  Result := DogColors[Random(Length(DogColors))];
end;

procedure TForm1.FormCreate(Sender: TObject);
var
  i: Integer;
begin

  FList := TObjectList<TDogInfo>.Create(True); // Owns the objects

  // Populate with sample data
  for i := 1 to 1000 do
    FList.Add(
      TDogInfo.Create(GetRandomDogName, Random(15), GetRandomDogColor)
    );

  FFilteredList := FList;

  lvDisplay.Items.Count := FFilteredList.Count;
  lvDisplay.Invalidate;

end;

procedure TForm1.FormDestroy(Sender: TObject);
begin
  if FFilteredList <> FList then
    FreeAndNil(FFilteredList);
  FreeAndNil(FList);
end;

这个想法是列表视图控件总是显示FFilteredList,它要么指向与 相同的对象实例FList,要么指向它的过滤(或排序)版本:

// The list view's OnData event handler
procedure TForm1.lvDisplayData(Sender: TObject; Item: TListItem);
begin

  if FFilteredList = nil then
    Exit;

  if not InRange(Item.Index, 0, FFilteredList.Count - 1) then
    Exit;

  Item.Caption := FFilteredList[Item.Index].Name;
  Item.SubItems.Add(FFilteredList[Item.Index].Age.ToString);
  Item.SubItems.Add(FFilteredList[Item.Index].Color);

end;

// The edit control's OnChange handler
procedure TForm1.eFilterChange(Sender: TObject);
var
  i: Integer;
begin

  if string(eFilter.Text).IsEmpty then // no filter, display all items
  begin
    if FFilteredList <> FList then
    begin
      FreeAndNil(FFilteredList);
      FFilteredList := FList;
    end;
  end
  else
  begin
    if (FFilteredList = nil) or (FFilteredList = FList) then
      FFilteredList := TObjectList<TDogInfo>.Create(False); // doesn't own the objects
    FFilteredList.Clear;
    for i := 0 to FList.Count - 1 do
      if FList[i].Matches(eFilter.Text) then
        FFilteredList.Add(FList[i]);
  end;

  lvDisplay.Items.Count := FFilteredList.Count;
  lvDisplay.Invalidate;

end;

结果:

填充列表的屏幕截图,没有过滤

过滤列表截图

请注意,每只狗始终只有一个内存对象,因此如果重命名狗,更改将反映在列表视图中,无论是否过滤。(但不要忘记使其无效!)

于 2019-09-24T06:23:50.170 回答
8

VCL 列表框的项目, API 中的列表框,没有任何可见性属性。不显示项目的唯一选择是删除它。

但是,您可以在根本没有项目的虚拟模式下使用控件。您决定要保留哪些数据,要显示什么。这是LBS_NODATAAPI 中的窗口样式。在 VCL 中,将style属性设置为lbVirtual.

下面是极其简化的示例。

让我们保留一组记录,每个虚拟项目一条记录。

type
  TListItem = record
    FileName: string;
    Visible: Boolean;
  end;

  TListItems = array of TListItem;

您可以根据需要扩展字段。我补充说,可见性是该问题的主要关注点之一。您可能会添加一些代表原始名称的内容,以便您知道已更改的名称等。

每个列表框有一个数组。此示例包含一个列表框。

var
  ListItems: TListItems;

最好让它成为一个字段,这仅用于演示。

所需单位。

uses
  ioutils, types;

表单创建时的一些初始化。清空过滤器编辑。相应地设置列表框样式。填写一些文件名。所有项目将在启动时可见。

procedure TForm1.FormCreate(Sender: TObject);
var
  ListFiles: TStringDynArray;
  i: Integer;
begin
  ListFiles := ioutils.TDirectory.GetFiles(TDirectory.GetCurrentDirectory);

  SetLength(ListItems, Length(ListFiles));
  for i := 0 to High(ListItems) do begin
    ListItems[i].FileName := ListFiles[i];
    ListItems[i].Visible := True;
  end;

  ListBox1.Style := lbVirtual;
  ListBox1.Count := Length(ListFiles);

  Edit1.Text := '';
end;

在虚拟模式下,列表框只对Count属性感兴趣。这将安排显示多少项目,相应地安排可滚动区域。

这是过滤器部分,这是区分大小写的。

procedure TForm1.Edit1Change(Sender: TObject);
var
  Text: string;
  Cnt: Integer;
  i: Integer;
begin
  Text := Edit1.Text;
  if Text = '' then begin
    for i := 0 to High(ListItems) do
      ListItems[i].Visible := True;
    Cnt := Length(ListItems);
  end else begin
    Cnt := 0;
    for i := 0 to High(ListItems) do begin
      ListItems[i].Visible := Pos(Text, ListItems[i].FileName) > 0;
      if ListItems[i].Visible then
        Inc(Cnt);
    end;
  end;
  ListBox1.Count := Cnt;
end;

编辑中的特殊情况OnChange是文本为空时。然后将显示所有项目。否则代码来自问题。这里我们还保留了可见项的总数,以便我们可以相应地更新列表框。

现在唯一有趣的部分是列表框需要数据。

procedure TForm1.ListBox1Data(Control: TWinControl; Index: Integer;
  var Data: string);
var
  VisibleIndex: Integer;
  i: Integer;
begin
  VisibleIndex := -1;
  for i := 0 to High(ListItems) do begin
    if ListItems[i].Visible then
      Inc(VisibleIndex);
    if VisibleIndex = Index then begin
      Data := ListItems[i].FileName;
      Break;
    end;
  end;
end;

这里发生的是列表框需要一个项目来显示提供它的索引。我们遍历主列表计数可见项目以找出与该索引匹配的项目,并提供其文本。

于 2019-09-24T00:14:43.007 回答