VirtualStringTree - 在使用对象时更正/处理子节点/子节点的正确方法?

时间:2011-11-04 06:03:41

标签: delphi delphi-2010 virtualtreeview tvirtualstringtree

我正在使用Delphi2010并尝试使用VirtualStringTree。

我一直试图让它与对象一起工作,并且在我遵循Philipp Frenzel的Virtual TreeView教程之前没有运气,我在soft-gems.net网站上找到了该教程。到目前为止我提出的工作有效,但我认为我没有正确处理子节点(即子节点)。

我唯一能够工作的是为每个孩子再次链接整个对象,然后只显示我需要的字段 - 但它只是感觉错了。

建议/反馈非常感谢。


我有一些对象列表,我正在尝试与VirtualStringTree连接,我正在尝试实现这样的目标,其中一个字段将作为父项的标签,其余字段显示为子节点。

  • 罗伯特莱恩
    • 35
    • 洛杉矶
    • 深色
  • Jane Doe
    • 19
    • 丹佛
    • 红发

这是我的课程设置方式。

type
  PTreeData = ^TTreeData;
  TTreeData = record
    FObject : TObject;
  end;

  TCustomerNode = class(TObject)
  private
    fName: string;
    fSex: string;
    fAge: integer;
    fHair: string;
    //...
  public
    property Name: string read fName write fName;
    //...
  end;

填充对象后,我将它们添加到另一个基于TList的类(CustomerObjectList),如下所述。

这是我将VirtualStringTree与我的对象列表连接的地方

procedure TfrmMain.btnLoadDataClick(Sender: TObject);
var
  i, j : integer;
  CustomerDataObject: TCustomerNode;
  RootXNode, XNode: PVirtualNode;
  Data: PTreeData;
begin
  vstree.NodeDataSize := SizeOf( TTreeData );

  vstree.BeginUpdate;
  for i := 0 to CustomerObjectList.Count - 1 do
  begin
    CustomerDataObject := CustomerObjectList[i];

    //load data for parent node
    RootXNode := vstree.AddChild(nil);
    Data  := vstree.GetNodeData(RootXNode);
    Data^.FObject:= PINodeSource;

    //now add children for rest of fields
    //Isn't there a better way to do this?
    for j := 1 to NUMBERofFIELDS -1 do  //first field is label for parent so -1
    begin
      XNode := vstree.AddChild(RootXNode);
      Data  := vstree.GetNodeData(XNode);
      Data^.FObject:= PINodeSource;
    end;

  end;
  vstree.EndUpdate; 
end;    

procedure TfrmMain.vstreeGetText(Sender: TBaseVirtualTree; Node: PVirtualNode;
 Column: TColumnIndex; TextType: TVSTTextType; var CellText: string);
var
  Data : PTreeData;
begin
  Data := vstObjects.GetNodeData(Node);
  ////if Node.ChildCount  = 0 then //edit - oops typo!
  if Node.ChildCount  > 0 then
  begin
    CellText := TCustomerNode(Data.FObject).Name;
    exit;
  end;

  //handle childnodes
  case Node.Index of
    0:  CellText := TCustomerNode(Data.FObject).Sex;
    1:  CellText := IntToStr(TCustomerNode(Data.FObject).Age);
    2:  CellText := TCustomerNode(Data.FObject).Hair;
    3:  CellText := TCustomerNode(Data.FObject).City;
  end;  
end;

2 个答案:

答案 0 :(得分:7)

您不需要将所有数据加载到树中。您可以使用'虚拟'这棵树这是我将如何做到的。

将树的RootNodeCount设置为CustomerObjectList中的记录数:

vstree.RootNodeCount := CustomerObjectList.Count;

然后,在OnInitChildren事件中,将0级节点的子计数设置为您要显示的属性数:

procedure TfrmMain.vstreeInitNode(Sender: TBaseVirtualTree; ParentNode, Node: PVirtualNode; var InitialStates: TVirtualNodeInitStates);
begin
  if Sender.GetNodeLevel(Node) = 0 then
  begin
    Sender.ChildCount[Node] := 4;

    // Comment this out if you don't want the nodes to be initially expanded
    Sender.Expanded[Node] := TRUE;
  end;
end;

现在,只需在OnGetText事件中检索正确的数据。

procedure TfrmMain.vstreeGetText(Sender: TBaseVirtualTree;
  Node: PVirtualNode; Column: TColumnIndex; TextType: TVSTTextType;
  var CellText: string);
begin
  if Column <= 0 then
  begin
    if Sender.GetNodeLevel(Node) = 0 then
      CellText := CustomerObjectList[Node.Index].Name else
    if Sender.GetNodeLevel(Node) = 1 then
    begin
      case Node.Index of
        0: CellText := CustomerObjectList[Parent.Node.Index].Sex;
        1: CellText := CustomerObjectList[Parent.Node.Index].Age;
        2: CellText := CustomerObjectList[Parent.Node.Index].Hair;
        3: CellText := CustomerObjectList[Parent.Node.Index].City;
      end; // of case
     end;
end;

为了以防万一,您可能想要添加一些范围检查。

希望这有帮助。

答案 1 :(得分:1)

  

我唯一能够工作的是连接整个   再次为每个孩子提出反对,然后只显示我需要的字段    - 但感觉很错误......建议/反馈非常赞赏。

你必须这样做,因为你正在混合水平。

您的树设置为期望每个级别的对象实例。但是,实例列表只包含主级别的对象和子级别的对象属性。

出于所有意图和目的,这没有任何问题。

如果你真的想避免它,你将不得不使用复合模式,其中一个对象拥有一个属性列表,这些属性本身又是对象。会工作,但也会让你的课程变得更加混乱。尽管您始终可以通过使用索引属性来提供对拥有属性的访问。

TSomeObject = class(TObject)
private
  MyObjects: TList<TSomeObject>;
protected
  function GetStringProperty(const aIndex: Integer): string;
public
  property Name: string index 1 read GetStringProperty;
end;

这称为实体属性值模型。 (The EAV/CR Model of Data Representation)它提供了灵活性但确实有缺点,特别是如果您以类似的方式构建数据库。