ORDENANDO LOS ELEMENTOS DE LA LISTA
El objeto ListView dispone de dos tipos de ordenación parecidos. Por un lado tenemos la ordenación mediante la función CustomSort:
function CustomSort( SortProc: TLVCompare; lParam: Longint ): Boolean;
Esta función toma como primer parámetro la dirección de una función CALLBACK encargada de establecer los parámetros de la ordenación. Veamos un ejemplo de ordenación ascendente por la columna ARTÍCULO (la segunda):
ListView.CustomSort( @Ordenacion, 0 );
Donde la función de ordenación sería la siguiente:
function Ordenacion( Item1, Item2: TListItem; ParamSort: Integer ): integer; stdcall;
begin
Result := CompareText( Item1.SubItems[1], Item2.SubItems[1] );
end;
Esta función compara los items 1 y 2 y le devuelve el resultado a CustomSort. Si queremos ordenarla descendentemente sería así:
function Ordenacion( Item1, Item2: TListItem; ParamSort: Integer ): integer; stdcall;
begin
Result := -CompareText( Item1.SubItems[1], Item2.SubItems[1] );
end;
Por otro lado tenemos la función AlphaSort la cual no necesita una función CALLBACK porque para eso tenemos el evento OnCompare. Se haría de la siguiente manera:
ListView.AlphaSort;
y en el evento OnCompare del ListView:
procedure TFormulario.ListViewCompare( Sender: TObject; Item1, Item2: TListItem;
Data: Integer; var Compare: Integer );
begin
Compare := CompareText( Item1.SubItems[1], Item2.SubItems[1] );
end;
Prácticamente es parecido a CustomSort (ya que un ListView desciende del componente CustomListView).
UTILIZANDO LA PRIMERA COLUMNA PARA CASOS ESPECIALES
Una de las cosas que se pueden meter en primera columna es un CheckBox para que el usuario marque filas. Antes de eso vamos a volver a mostrar la primera columna (Nada) a 70 de ancho y le cambiamos el nombre a Seleccionado. Por último activamos en las propiedades del ListView el campo CheckBoxes.
Al ejecutar el programa y dar de alta filas en el listado veremos que por cada elemento aparece un CheckBox a la izquierda para poder marcarlo. ¿Cómo sabemos metiante código los elementos que han sido marcados? Con la propiedad Checked de cada Item.
Veamos un ejemplo que comprueba los artículos marcados con CheckBox, los vuelca a un StringList y los saca a pantalla:
var
Seleccionados: TStringList;
i: Integer;
begin
Seleccionados := TStringList.Create;
for i := 0 to ListView.Items.Count - 1 do
if ListView.Items[i].Checked then
Seleccionados.Add( ListView.Items[i].SubItems[1] );
ShowMessage( Seleccionados.Text );
Seleccionados.Free;
end;
Otra utilidad muy interesante que se le puede dar a la primera columna es asociarle una imagen a cada fila. Para ello hay que añadir al formulario el componente ImageList que encuentra en la pestaña Win32.
Después añadimos imágenes a la lista y asociamos las propiedades LargeImages, StateImages y SmallImages con el componente ImageList. Con sólo hacer eso todas las filas tendrán a la izquierda la primera imagen de la lista de imágenes. Para cambiar la imagen en una fila determinada se hace:
ListView.Items[2].ImageIndex := 2;
o bien cambiamos sólo la fila seleccionada por el usuario:
if ListView.Selected <> nil then
ListView.Selected.ImageIndex := 1;
ACTIVANDO LA SELECCIÓN AUTOMÁTICA
Si activamos en el componente ListView la propiedad HotTrack y pasamos el puntero del ratón por las filas veremos como se iluminan en otro color, quedando seleccionadas fijamente si permanecemos con el ratón sobre las mismas.
Esto puede ser de utilidad para crear programas visualizadores de imágenes que muestren fotografías con sólo posicionarse con el ratón encima del nombre de la foto JPG. O para crear un listado de URL para nuestras páginas web preferidas.
También se pueden activar las propiedades:
htHandPoint -> Cambia el puntero del ratón a una mano cuando se para por encima de las filas.
htUnderlineCold -> Subraya la fila que se va a seleccionar
htUnderlineHold -> Subraya la fila que hay seleccionada
En el próximo artículo vamos a ver como reprogramar el dibujado de las filas en tiempo real.
Pruebas realizadas en Delphi 7.
1 comentario:
Hola, está muy bueno el tuto sobre los ListView ya que ahora mismo estoy trabajando con ella, ¿es posible utilizar la primera columna para que muestre el número del item o usarlo como una columna autoincrementable?, por ejemplo:
No | Apellido | Nombre
1. | Leonor | Fernando
2. | Leonor2 | Fernando2
3. | leonor3 | Fernando3
Saludos.
Publicar un comentario