Wx::Perl::ListCtrl - a sensible API for Wx::ListCtrl


Wx-Perl-ListCtrl documentation Contained in the Wx-Perl-ListCtrl distribution.

Index


Code Index:

NAME

Top

Wx::Perl::ListCtrl - a sensible API for Wx::ListCtrl

SYNOPSIS

Top

    use Wx::Perl::ListCtrl;

    my $lc = Wx::Perl::ListCtrl->new( $parent, -1 );

    # add columns...

    # get/set item text easily
    $lc->InsertStringItem( 0, 'dummy' );
    $lc->SetItemText( 0, 0, 'row 0, col 0' );
    $lc->SetItemText( 0, 1, 'row 0, col 1' );
    $lc->GetItemText( 0, 1 ) # 'row 0, col 1'

    # use structured data, not plain integers
    $lc->SetItemData( 0, { complex =>1, data => 2 } );
    my $data = $lc->GetItemData( 0 );

    # sensible way of getting the selection
    my $selection = $lc->GetSelection;   # single selection
    my @selections = $lc->GetSelections; # multiple selections

DESCRIPTION

Top

The Wx::ListCtrl API is terrible. This module goes further than Wx::ListView in providing a sane api for Wx::ListCtrl.

This module is not a drop-in replacement for Wx::ListCtrl and Wx::ListView .

Wx::Perl::ListCtrl derives from Wx::ListView, however some of Wx::ListView/Wx::ListCtrl methods are overridden with more sensible, and sometimes API-incompatible, implementations.

METHODS

Top

GetSelection

    my $selection = $lc->GetSelection;

Returns the single selected line. Only works with a single-selection list control.

GetSelections

    my @selections = $lc->GetSelections;

Returns a list with all the selected lines. Only works with a multi-selection list control.

GetItemText

    my $text = $lc->GetItemText( $row, $col );

WARNING: incompatible with Wx::ListCtrl . Returns the text of the given item.

SetItemText

    $lc->SetItemText( $row, $col, 'Text' );

WARNING: incompatible with Wx::ListCtrl . Sets the text of the given item.

SetItemData

    $lc->SetItemData( $item, { complex => [ qw(data is allowed) ] } );

Sets the client data for the given row. Complex data structures are allowed. Setting the data to undef deletes the data for the given row.

GetItemData

    my $data = $lc->GetItemData( $data );

Retrieves the data set with $lc-SetItemData>.

BUGS

Top

Calling SetItemData too many times will crash after about 2^31 per-object calls (with 32 bit integers).

AUTHOR

Top

Mattia Barbon <mbarbon@cpan.org>

LICENSE

Top

Copyright (c) 2005, 2007 Mattia Barbon <mbarbon@cpan.org>

This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself


Wx-Perl-ListCtrl documentation Contained in the Wx-Perl-ListCtrl distribution.
package Wx::Perl::ListCtrl;

use strict;

our $VERSION = '0.03';

use Wx qw(:listctrl);
use base 'Wx::ListView';

# assume a 4 byte long value giving a max of 2147483647 and set the
# max index a little below.
sub _max_itemdata_idx() { 2147483640 }
sub _carp { require Carp; goto &Carp::carp; }

sub GetSelection {
    my $self = shift;

    _carp( "GetSelection must be used on single selection Wx::Perl::ListCtrl" )
      unless $self->GetWindowStyleFlag & wxLC_SINGLE_SEL;

    return $self->GetFirstSelected;
}

sub GetSelections {
    my $self = shift;

    _carp( "GetSelections must be used on multi selection Wx::Perl::ListCtrl" )
      if $self->GetWindowStyleFlag & wxLC_SINGLE_SEL;

    my $selection = $self->GetFirstSelected;

    return if $selection == -1;

    my @selections = ( $selection );

    while( ( $selection = $self->GetNextSelected( $selection ) ) != -1 ) {
        push @selections, $selection;
    }

    return @selections;
}

sub GetItemText {
    my( $self, $item, $col ) = @_; $col ||= 0;

    return $self->SUPER::GetItemText( $item ) if $col == 0;
    my $it = $self->GetItem( $item, $col );

    return $it ? $it->GetText : '';
}

*SetItemText = \&Wx::ListCtrl::SetItemString;

sub SetItemData {
    use integer;

    my( $self, $item, $data ) = @_;
    my $stash = $self->{_wx_data} ||= {};
    my $idx = sprintf "%u", $self->SUPER::GetItemData( $item ) || 0;

    unless( defined $data ) {
        delete $stash->{$idx};
        return;
    }

    unless( $idx ) {
        $idx = _get_new_idx( $self );
        # reset where $stash points
        $stash = $self->{_wx_data};
    }

    $stash->{$idx} = $data;

    $self->SUPER::SetItemData( $item, $idx );
}

sub _get_new_idx {
    use integer;

    my( $self ) = @_;
    my $idx = sprintf "%u", ++$self->{_wx_count};

    return $idx if $idx < _max_itemdata_idx();

    # reset stash and item data
    my $oldstash = $self->{_wx_data};
    $self->{_wx_count} = 0;
    my $newstash = {};
    for( my $item = $self->SUPER::GetNextItem( -1 );
         $item != -1;
         $item = $self->SUPER::GetNextItem( $item ) ) {
        my $oldindex = $self->SUPER::GetItemData( $item ) || 0;
        if( $oldindex && exists $oldstash->{$oldindex} ) {
            my $newindex = sprintf "%u", ++$self->{_wx_count};
            $newstash->{$newindex} = $oldstash->{$oldindex};
            $self->SUPER::SetItemData( $item, $newindex );
        }
    }
    $self->{_wx_data} = $newstash;

    return _get_new_idx( $self );
}

sub GetItemData {
    use integer;

    my( $self, $item ) = @_;
    my $stash = $self->{_wx_data};
    return unless $stash;

    return $stash->{$self->SUPER::GetItemData( $item ) || 0};
}

# overridden to correctly handle the custom item data,
# they do not change user-visible behaviour

sub DeleteAllItems {
    my $self = shift;
    my $ret = $self->SUPER::DeleteAllItems;

    if( $ret ) {
        delete $self->{_wx_data};
        delete $self->{_wx_count};
    }

    return $ret;
}

sub ClearAll {
    my $self = shift;

    $self->SUPER::ClearAll;

    delete $self->{_wx_data};
    delete $self->{_wx_count};
}

sub DeleteItem {
    my( $self, $item ) = @_;
    my $key = $self->SUPER::GetItemData( $item );
    my $ret = $self->SUPER::DeleteItem( $item );

    delete $self->{_wx_data}{$key} if $ret;

    return $ret;
}

1;

__END__