Search code examples
delphianimationlistboxfiremonkeytlistbox

Animating the addition of a string to a ListBox in FireMonkey


The following code nicely animates adding a new string to the end of a ListBox

procedure TForm6.AddItem(s: string);
var
  l : TListBoxItem;
  OldHeight : Single;
begin
  l := TListBoxItem.Create(Self);
  l.Text := s;
  OldHeight := l.Height;
  l.Height := 0;
  l.Parent := ListBox1;
  l.Opacity := 0;
  l.AnimateFloat('height', OldHeight, 0.5);
  l.AnimateFloat('Opacity', 1, 0.5);
end;

The item expands and fades in. However I want to be able to add the string into an arbitrary location in the ListBox - actually at the current ItemIndex. Does anyone know how to do this?


Solution

  • To work around the fact that ListBox1.InsertObject and ListBox1.Items.Insert don't work you can do the following

    procedure TForm1.AddItem(s: string);
    var
      l : TListBoxItem;
      OldHeight : Single;
      I: Integer;
      index : integer;
    begin
      l := TListBoxItem.Create(nil);
      l.Text := s;
      OldHeight := l.Height;
      l.Height := 0;
      l.Opacity := 0;
      l.Index := 0;
      l.Parent := ListBox1;
    
      Index := Max(0, ListBox1.ItemIndex);
      for I := ListBox1.Count - 1 downto Index + 1 do
      begin
        ListBox1.Exchange(ListBox1.ItemByIndex(i), ListBox1.ItemByIndex(i-1));
      end;
      ListBox1.ItemIndex := Index;
      l.AnimateFloat('height', OldHeight, 0.5);
      l.AnimateFloat('Opacity', 1, 0.5);
    end;
    

    but is a bit ridiculous. It (eventually) adds the string in position 0 if there is no item selected, otherwise adds it before the selected item. This solution reminds me too much of Bubble Sort. You will need to add the math unit to your uses clause for the max function to work.

    This does indeed seem to be a bug in FireMonkey (check Quality Central #102122), However I suspect a future FireMonkey update will fix this. If anyone can see a better way of doing this....

    I've also made a movie about this for those who are interested, which illustrates things more clearly.