]> git.saurik.com Git - wxWidgets.git/blobdiff - src/generic/listctrl.cpp
Corrected wxScrolledWindow's OnSize behaviour.
[wxWidgets.git] / src / generic / listctrl.cpp
index 8ea3034af7b1661c2d396d09e03b976e0e62a7a8..d2f3b996759c4935f5632e7168f236075751a7a0 100644 (file)
@@ -8,6 +8,45 @@
 // Licence:     wxWindows licence
 /////////////////////////////////////////////////////////////////////////////
 
+/*
+   FIXME for virtual list controls
+
+  +1. clicking on the item with a mouse is awfully slow, what is going on?
+      note that selecting with keyboard seems to be much faster
+   => fixed HighlightAll() - iterating over 1000000 items *is* slow
+
+   2. background colour is wrong?
+ */
+
+/*
+   TODO for better virtual list control support:
+
+   1. less dumb line caching, we should cache at least all those visible
+      in the control itself and probably twice as many (we might also need to
+      cache the first one always for geometry calculations?)
+
+  +2. storing selections: we can't use an array to store the selected indices
+      like right now as selecting all in a control with 1000000 items is not
+      doable like this - instead, store selections as collection of individual
+      items and ranges
+
+   => wxSelectionStore
+
+   3. we need to implement searching/sorting somehow
+
+   4. the idea of storing the line index in the line itself is really stupid,
+      we shouldn't need it - but for this we have to get rid of all calles to
+      wxListLineData::GetFoo() and replace them with something like
+        if ( IsVirtual()
+            ... we have it ourselves ...
+        else
+            line->GetFoo();
+
+   => done
+
+   5. attributes support: we need OnGetItemAttr() as well!
+ */
+
 // ============================================================================
 // declarations
 // ============================================================================
     #pragma implementation "listctrlbase.h"
 #endif
 
-#if 0
-    #include "listctrl.old.cpp"
-#else
-
 // For compilers that support precompilation, includes "wx.h".
 #include "wx/wxprec.h"
 
@@ -37,7 +72,7 @@
 #include "wx/dcscreen.h"
 #include "wx/app.h"
 #include "wx/listctrl.h"
-#include "wx/generic/imaglist.h"
+#include "wx/imaglist.h"
 #include "wx/dynarray.h"
 
 #ifdef __WXGTK__
@@ -78,7 +113,7 @@ static const int SCROLL_UNIT_X = 15;
 static const int SCROLL_UNIT_Y = 15;
 
 // the spacing between the lines (in report mode)
-static const int LINE_SPACING = 1;
+static const int LINE_SPACING = 0;
 
 // extra margins around the text label
 static const int EXTRA_WIDTH = 3;
@@ -99,6 +134,76 @@ static const int WIDTH_COL_MIN = 10;
 // private classes
 // ============================================================================
 
+// ----------------------------------------------------------------------------
+// wxSelectionStore
+// ----------------------------------------------------------------------------
+
+int CMPFUNC_CONV wxSizeTCmpFn(size_t n1, size_t n2) { return n1 - n2; }
+
+WX_DEFINE_SORTED_EXPORTED_ARRAY(size_t, wxIndexArray);
+
+// this class is used to store the selected items in the virtual list control
+// (but it is not tied to list control and so can be used with other controls
+// such as wxListBox in wxUniv)
+//
+// the idea is to make it really smart later (i.e. store the selections as an
+// array of ranes + individual items) but, as I don't have time to do it now
+// (this would require writing code to merge/break ranges and much more) keep
+// it simple but define a clean interface to it which allows it to be made
+// smarter later
+class WXDLLEXPORT wxSelectionStore
+{
+public:
+    wxSelectionStore() : m_itemsSel(wxSizeTCmpFn) { Init(); }
+
+    // set the total number of items we handle
+    void SetItemCount(size_t count) { m_count = count; }
+
+    // special case of SetItemCount(0)
+    void Clear() { m_itemsSel.Clear(); m_count = 0; }
+
+    // must be called when a new item is inserted/added
+    void OnItemAdd(size_t item) { wxFAIL_MSG( _T("TODO") ); }
+
+    // must be called when an item is deleted
+    void OnItemDelete(size_t item);
+
+    // select one item, use SelectRange() insted if possible!
+    //
+    // returns true if the items selection really changed
+    bool SelectItem(size_t item, bool select = TRUE);
+
+    // select the range of items
+    void SelectRange(size_t itemFrom, size_t itemTo, bool select = TRUE);
+
+    // return true if the given item is selected
+    bool IsSelected(size_t item) const;
+
+    // return the total number of selected items
+    size_t GetSelectedCount() const
+    {
+        return m_defaultState ? m_count - m_itemsSel.GetCount()
+                              : m_itemsSel.GetCount();
+    }
+
+private:
+    // (re)init
+    void Init() { m_defaultState = FALSE; }
+
+    // the total number of items we handle
+    size_t m_count;
+
+    // the default state: normally, FALSE (i.e. off) but maybe set to TRUE if
+    // there are more selected items than non selected ones - this allows to
+    // handle selection of all items efficiently
+    bool m_defaultState;
+
+    // the array of items whose selection state is different from default
+    wxIndexArray m_itemsSel;
+
+    DECLARE_NO_COPY_CLASS(wxSelectionStore)
+};
+
 //-----------------------------------------------------------------------------
 //  wxListItemData (internal)
 //-----------------------------------------------------------------------------
@@ -142,7 +247,8 @@ public:
 
     void GetItem( wxListItem &info ) const;
 
-    wxListItemAttr *GetAttributes() const { return m_attr; }
+    void SetAttr(wxListItemAttr *attr) { m_attr = attr; }
+    wxListItemAttr *GetAttr() const { return m_attr; }
 
 public:
     // the item image or -1
@@ -237,17 +343,17 @@ public:
         wxRect m_rectIcon;
 
         // the part to be highlighted
-        wxRect m_rectHilight;
+        wxRect m_rectHighlight;
     } *m_gi;
 
     // is this item selected? [NB: not used in virtual mode]
-    bool m_hilighted;
+    bool m_highlighted;
 
     // back pointer to the list ctrl
     wxListMainWindow *m_owner;
 
 public:
-    wxListLineData( wxListMainWindow *owner, size_t line );
+    wxListLineData(wxListMainWindow *owner);
 
     ~wxListLineData() { delete m_gi; }
 
@@ -255,7 +361,7 @@ public:
     inline bool InReportView() const;
 
     // are we in virtual report mode?
-    inline bool IsVirtal() const;
+    inline bool IsVirtual() const;
 
     // these 2 methods shouldn't be called for report view controls, in that
     // case we determine our position/size ourselves
@@ -266,7 +372,12 @@ public:
     // remember the position this line appears at
     void SetPosition( int x, int y,  int window_width, int spacing );
 
-    long IsHit( int x, int y );
+    // wxListCtrl API
+
+    void SetImage( int image ) { SetImage(0, image); }
+    int GetImage() const { return GetImage(0); }
+    bool HasImage() const { return GetImage() != -1; }
+    bool HasText() const { return !GetText(0).empty(); }
 
     void SetItem( int index, const wxListItem &info );
     void GetItem( int index, wxListItem &info );
@@ -274,42 +385,29 @@ public:
     wxString GetText(int index) const;
     void SetText( int index, const wxString s );
 
-    void SetImage( int index, int image );
-    int GetImage( int index ) const;
-
-    // get the bound rect of this line
-    wxRect GetRect() const;
-
-    // get the bound rect of the label
-    wxRect GetLabelRect() const;
-
-    // get the bound rect of the items icon (only may be called if we do have
-    // an icon!)
-    wxRect GetIconRect() const;
-
-    // get the rect to be highlighted when the item has focus
-    wxRect GetHighlightRect() const;
-
-    // get the size of the total line rect
-    wxSize GetSize() const { return GetRect().GetSize(); }
+    wxListItemAttr *GetAttr() const;
+    void SetAttr(wxListItemAttr *attr);
 
     // return true if the highlighting really changed
-    bool Hilight( bool on );
-
-    void ReverseHilight();
+    bool Highlight( bool on );
 
-    // draw the line on the given DC
-    void Draw( wxDC *dc, int y = 0, int height = 0, bool highlighted = FALSE );
+    void ReverseHighlight();
 
-    bool IsHilighted() const
+    bool IsHighlighted() const
     {
-        wxASSERT_MSG( !IsVirtal(), _T("unexpected call to IsHilighted") );
+        wxASSERT_MSG( !IsVirtual(), _T("unexpected call to IsHighlighted") );
 
-        return m_hilighted;
+        return m_highlighted;
     }
 
-    // only for wxListMainWindow::CacheLineData()
-    void SetLineIndex(size_t line) { m_lineIndex = line; }
+    // draw the line on the given DC in icon/list mode
+    void Draw( wxDC *dc );
+
+    // the same in report mode
+    void DrawInReportMode( wxDC *dc,
+                           const wxRect& rect,
+                           const wxRect& rectHL,
+                           bool highlighted );
 
 private:
     // set the line to contain num items (only can be > 1 in report mode)
@@ -322,13 +420,14 @@ private:
                        const wxListItemAttr *attr,
                        const wxColour& colText,
                        const wxFont& font,
-                       bool hilight);
+                       bool highlight);
 
-    // the index of this line (only used in report mode)
-    size_t m_lineIndex;
+    // these are only used by GetImage/SetImage above, we don't support images
+    // with subitems at the public API level yet
+    void SetImage( int index, int image );
+    int GetImage( int index ) const;
 };
 
-
 WX_DECLARE_EXPORTED_OBJARRAY(wxListLineData, wxListLineDataArray);
 #include "wx/arrimpl.cpp"
 WX_DEFINE_OBJARRAY(wxListLineDataArray);
@@ -453,27 +552,68 @@ public:
     // return true if this is a virtual list control
     bool IsVirtual() const { return HasFlag(wxLC_VIRTUAL); }
 
+    // return true if the control is in report mode
+    bool InReportView() const { return HasFlag(wxLC_REPORT); }
+
+    // return true if we are in single selection mode, false if multi sel
+    bool IsSingleSel() const { return HasFlag(wxLC_SINGLE_SEL); }
+
     // do we have a header window?
     bool HasHeader() const
         { return HasFlag(wxLC_REPORT) && !HasFlag(wxLC_NO_HEADER); }
 
-    void HilightAll( bool on );
+    void HighlightAll( bool on );
 
     // all these functions only do something if the line is currently visible
 
     // change the line "selected" state, return TRUE if it really changed
-    bool HilightLine( size_t line, bool hilight = TRUE);
+    bool HighlightLine( size_t line, bool highlight = TRUE);
+
+    // as HighlightLine() but do it for the range of lines: this is incredibly
+    // more efficient for virtual list controls!
+    //
+    // NB: unlike HighlightLine() this one does refresh the lines on screen
+    void HighlightLines( size_t lineFrom, size_t lineTo, bool on = TRUE );
 
     // toggle the line state and refresh it
-    void ReverseHilight( size_t line )
-        { HilightLine(line, !IsHilighted(line)); RefreshLine(line); }
+    void ReverseHighlight( size_t line )
+        { HighlightLine(line, !IsHighlighted(line)); RefreshLine(line); }
+
+    // return true if the line is highlighted
+    bool IsHighlighted(size_t line) const;
 
     // refresh one or several lines at once
     void RefreshLine( size_t line );
     void RefreshLines( size_t lineFrom, size_t lineTo );
 
-    // return true if the line is highlighted
-    bool IsHilighted(size_t line) const;
+    // refresh all lines below the given one: the difference with
+    // RefreshLines() is that the index here might not be a valid one (happens
+    // when the last line is deleted)
+    void RefreshAfter( size_t lineFrom );
+
+    // the methods which are forwarded to wxListLineData itself in list/icon
+    // modes but are here because the lines don't store their positions in the
+    // report mode
+
+    // get the bound rect for the entire line
+    wxRect GetLineRect(size_t line) const;
+
+    // get the bound rect of the label
+    wxRect GetLineLabelRect(size_t line) const;
+
+    // get the bound rect of the items icon (only may be called if we do have
+    // an icon!)
+    wxRect GetLineIconRect(size_t line) const;
+
+    // get the rect to be highlighted when the item has focus
+    wxRect GetLineHighlightRect(size_t line) const;
+
+    // get the size of the total line rect
+    wxSize GetLineSize(size_t line) const
+        { return GetLineRect(line).GetSize(); }
+
+    // return the hit code for the corresponding position (in this line)
+    long HitTestLine(size_t line, int x, int y) const;
 
     void EditLabel( long item );
     void OnRenameTimer();
@@ -489,14 +629,13 @@ public:
     void OnKeyDown( wxKeyEvent &event );
     void OnSetFocus( wxFocusEvent &event );
     void OnKillFocus( wxFocusEvent &event );
-    void OnSize( wxSizeEvent &event );
     void OnScroll(wxScrollWinEvent& event) ;
 
     void OnPaint( wxPaintEvent &event );
 
     void DrawImage( int index, wxDC *dc, int x, int y );
-    void GetImageSize( int index, int &width, int &height );
-    int GetTextLength( const wxString &s );
+    void GetImageSize( int index, int &width, int &height ) const;
+    int GetTextLength( const wxString &s ) const;
 
     void SetImageList( wxImageList *imageList, int which );
     void SetItemSpacing( int spacing, bool isSmall = FALSE );
@@ -511,7 +650,7 @@ public:
     // returns the sum of the heights of all columns
     int GetHeaderWidth() const;
 
-    int GetCountPerPage() { return m_linesPerPage; }
+    int GetCountPerPage() const;
 
     void SetItem( wxListItem &item );
     void GetItem( wxListItem &item );
@@ -522,7 +661,10 @@ public:
     int GetSelectedItemCount();
 
     // set the scrollbars and update the positions of the items
-    void CalculatePositions();
+    void RecalculatePositions();
+
+    // refresh the window and the header
+    void RefreshAll();
 
     long GetNextItem( long item, int geometry, int state );
     void DeleteItem( long index );
@@ -541,6 +683,7 @@ public:
     bool IsEmpty() const { return GetItemCount() == 0; }
     void SetItemCount(long count);
 
+    void ResetCurrent() { m_current = (size_t)-1; }
     bool HasCurrent() const { return m_current != (size_t)-1; }
 
     // send out a wxListEvent
@@ -548,8 +691,16 @@ public:
                      wxEventType command,
                      wxPoint point = wxDefaultPosition );
 
-    // called by wxListCtrl when its font changes
-    void OnFontChange() { m_lineHeight = 0; }
+    // override base class virtual to reset m_lineHeight when the font changes
+    virtual bool SetFont(const wxFont& font)
+    {
+        if ( !wxScrolledWindow::SetFont(font) )
+            return FALSE;
+
+        m_lineHeight = 0;
+
+        return TRUE;
+    }
 
     // these are for wxListLineData usage only
 
@@ -589,8 +740,8 @@ public:
     // call
     bool                 m_dirty;
 
-    wxBrush             *m_hilightBrush;
-    wxColour            *m_hilightColour;
+    wxBrush             *m_highlightBrush;
+    wxColour            *m_highlightColour;
     int                  m_xScroll,
                          m_yScroll;
     wxImageList         *m_small_image_list;
@@ -613,15 +764,11 @@ public:
 
 protected:
     // the total count of items in a virtual list control
-    long m_countVirt;
-
-    // the first and last lines being shown on screen right now (inclusive)
-    size_t m_lineFrom,
-           m_lineTo;
+    size_t m_countVirt;
 
-    // the array containing the indices of all selected items, only used in
-    // virtual controls
-    wxArrayInt m_selections;
+    // the object maintaining the items selection state, only used in virtual
+    // controls
+    wxSelectionStore m_selStore;
 
     // common part of all ctors
     void Init();
@@ -644,16 +791,28 @@ protected:
         return &m_lines[n];
     }
 
-    // get the first line: this one is special as we have it even in virtual
-    // list control (it is useful to cache it as we use it for measuring, hit
-    // testing &c)
-    wxListLineData *GetFirstLine() const;
+    // get a dummy line which can be used for geometry calculations and such:
+    // you must use GetLine() if you want to really draw the line
+    wxListLineData *GetDummyLine() const;
 
     // cache the line data of the n-th line in m_lines[0]
     void CacheLineData(size_t line);
 
-    // update m_lineFrom/To
-    void UpdateShownLinesRange();
+    // get the range of visible lines
+    void GetVisibleLinesRange(size_t *from, size_t *to);
+
+    // force us to recalculate the range of visible lines
+    void ResetVisibleLinesRange() { m_lineFrom = (size_t)-1; }
+
+    // get the colour to be used for drawing the rules
+    wxColour GetRuleColour() const
+    {
+#ifdef __WXMAC__
+        return *wxWHITE;
+#else
+        return wxSystemSettings::GetSystemColour(wxSYS_COLOUR_3DLIGHT);
+#endif
+    }
 
 private:
     // initialize the current item if needed
@@ -671,6 +830,12 @@ private:
     // the total header width or 0 if not calculated yet
     wxCoord m_headerWidth;
 
+    // the first and last lines being shown on screen right now (inclusive),
+    // both may be -1 if they must be calculated so never access them directly:
+    // use GetVisibleLinesRange() above instead
+    size_t m_lineFrom,
+           m_lineTo;
+
     DECLARE_DYNAMIC_CLASS(wxListMainWindow);
     DECLARE_EVENT_TABLE()
 };
@@ -679,6 +844,141 @@ private:
 // implementation
 // ============================================================================
 
+// ----------------------------------------------------------------------------
+// wxSelectionStore
+// ----------------------------------------------------------------------------
+
+bool wxSelectionStore::IsSelected(size_t item) const
+{
+    bool isSel = m_itemsSel.Index(item) != wxNOT_FOUND;
+
+    // if the default state is to be selected, being in m_itemsSel means that
+    // the item is not selected, so we have to inverse the logic
+    return m_defaultState ? !isSel : isSel;
+}
+
+bool wxSelectionStore::SelectItem(size_t item, bool select)
+{
+    // search for the item ourselves as like this we get the index where to
+    // insert it later if needed, so we do only one search in the array instead
+    // of two (adding item to a sorted array requires a search)
+    size_t index = m_itemsSel.IndexForInsert(item);
+    bool isSel = index < m_itemsSel.GetCount() && m_itemsSel[index] == item;
+
+    if ( select != m_defaultState )
+    {
+        if ( !isSel )
+        {
+            m_itemsSel.AddAt(item, index);
+
+            return TRUE;
+        }
+    }
+    else // reset to default state
+    {
+        if ( isSel )
+        {
+            m_itemsSel.RemoveAt(index);
+            return TRUE;
+        }
+    }
+
+    return FALSE;
+}
+
+void wxSelectionStore::SelectRange(size_t itemFrom, size_t itemTo, bool select)
+{
+    wxASSERT_MSG( itemFrom <= itemTo, _T("should be in order") );
+
+    // are we going to have more [un]selected items than the other ones?
+    if ( itemTo - itemFrom > m_count / 2 )
+    {
+        if ( select != m_defaultState )
+        {
+            // the default state now becomes the same as 'select'
+            m_defaultState = select;
+
+            // so all the old selections (which had state select) shouldn't be
+            // selected any more, but all the other ones should
+            wxIndexArray selOld = m_itemsSel;
+            m_itemsSel.Empty();
+
+            // TODO: it should be possible to optimize the searches a bit
+            //       knowing the possible range
+
+            size_t item;
+            for ( item = 0; item < itemFrom; item++ )
+            {
+                if ( selOld.Index(item) == wxNOT_FOUND )
+                    m_itemsSel.Add(item);
+            }
+
+            for ( item = itemTo + 1; item < m_count; item++ )
+            {
+                if ( selOld.Index(item) == wxNOT_FOUND )
+                    m_itemsSel.Add(item);
+            }
+        }
+        else // select == m_defaultState
+        {
+            // get the inclusive range of items between itemFrom and itemTo
+            size_t count = m_itemsSel.GetCount(),
+                   start = m_itemsSel.IndexForInsert(itemFrom),
+                   end = m_itemsSel.IndexForInsert(itemTo);
+
+            if ( start == count || m_itemsSel[start] < itemFrom )
+            {
+                start++;
+            }
+
+            if ( end == count || m_itemsSel[end] > itemTo )
+            {
+                end--;
+            }
+
+            if ( start <= end )
+            {
+                // delete all of them (from end to avoid changing indices)
+                for ( int i = end; i >= (int)start; i-- )
+                {
+                    m_itemsSel.RemoveAt(i);
+                }
+            }
+        }
+    }
+    else // "few" items change state
+    {
+        // just add the items to the selection
+        for ( size_t item = itemFrom; item <= itemTo; item++ )
+        {
+            SelectItem(item, select);
+        }
+    }
+}
+
+void wxSelectionStore::OnItemDelete(size_t item)
+{
+    size_t count = m_itemsSel.GetCount(),
+           i = m_itemsSel.IndexForInsert(item);
+
+    if ( i < count && m_itemsSel[i] == item )
+    {
+        // this item itself was in m_itemsSel, remove it from there
+        m_itemsSel.RemoveAt(i);
+
+        count--;
+    }
+
+    // and adjust the index of all which follow it
+    while ( i < count )
+    {
+        // all following elements must be greater than the one we deleted
+        wxASSERT_MSG( m_itemsSel[i] > item, _T("logic error") );
+
+        m_itemsSel[i++]--;
+    }
+}
+
 //-----------------------------------------------------------------------------
 //  wxListItemData
 //-----------------------------------------------------------------------------
@@ -903,7 +1203,7 @@ int wxListHeaderData::GetFormat() const
 
 inline int wxListLineData::GetMode() const
 {
-    return m_owner->GetListCtrl()->GetWindowStyleFlag() & wxLC_MODE_MASK;
+    return m_owner->GetListCtrl()->GetWindowStyleFlag() & wxLC_MASK_TYPE;
 }
 
 inline bool wxListLineData::InReportView() const
@@ -911,18 +1211,16 @@ inline bool wxListLineData::InReportView() const
     return m_owner->HasFlag(wxLC_REPORT);
 }
 
-inline bool wxListLineData::IsVirtal() const
+inline bool wxListLineData::IsVirtual() const
 {
     return m_owner->IsVirtual();
 }
 
-wxListLineData::wxListLineData( wxListMainWindow *owner, size_t line )
+wxListLineData::wxListLineData( wxListMainWindow *owner )
 {
     m_owner = owner;
     m_items.DeleteContents( TRUE );
 
-    SetLineIndex(line);
-
     if ( InReportView() )
     {
         m_gi = NULL;
@@ -932,64 +1230,11 @@ wxListLineData::wxListLineData( wxListMainWindow *owner, size_t line )
         m_gi = new GeometryInfo;
     }
 
-    m_hilighted = FALSE;
+    m_highlighted = FALSE;
 
     InitItems( GetMode() == wxLC_REPORT ? m_owner->GetColumnCount() : 1 );
 }
 
-wxRect wxListLineData::GetRect() const
-{
-    if ( !InReportView() )
-        return m_gi->m_rectAll;
-
-    wxRect rect;
-    rect.x = HEADER_OFFSET_X;
-    rect.y = m_owner->GetLineY(m_lineIndex);
-    rect.width = m_owner->GetHeaderWidth();
-    rect.height = m_owner->GetLineHeight();
-
-    return rect;
-}
-
-wxRect wxListLineData::GetLabelRect() const
-{
-    if ( !InReportView() )
-        return m_gi->m_rectLabel;
-
-    wxRect rect;
-    rect.x = HEADER_OFFSET_X;
-    rect.y = m_owner->GetLineY(m_lineIndex);
-    rect.width = m_owner->GetColumnWidth(0);
-    rect.height = m_owner->GetLineHeight();
-
-    return rect;
-}
-
-wxRect wxListLineData::GetIconRect() const
-{
-    if ( !InReportView() )
-        return m_gi->m_rectIcon;
-
-    wxRect rect;
-
-    wxListItemDataList::Node *node = m_items.GetFirst();
-    wxCHECK_MSG( node, rect, _T("no subitems at all??") );
-
-    wxListItemData *item = node->GetData();
-    wxASSERT_MSG( item->HasImage(), _T("GetIconRect() called but no image") );
-
-    rect.x = HEADER_OFFSET_X;
-    rect.y = m_owner->GetLineY(m_lineIndex);
-    m_owner->GetImageSize(item->GetImage(), rect.width, rect.height);
-
-    return rect;
-}
-
-wxRect wxListLineData::GetHighlightRect() const
-{
-    return InReportView() ? GetRect() : m_gi->m_rectHilight;
-}
-
 void wxListLineData::CalculateSize( wxDC *dc, int spacing )
 {
     wxListItemDataList::Node *node = m_items.GetFirst();
@@ -1044,13 +1289,13 @@ void wxListLineData::CalculateSize( wxDC *dc, int spacing )
 
                 if ( item->HasText() )
                 {
-                    m_gi->m_rectHilight.width = m_gi->m_rectLabel.width;
-                    m_gi->m_rectHilight.height = m_gi->m_rectLabel.height;
+                    m_gi->m_rectHighlight.width = m_gi->m_rectLabel.width;
+                    m_gi->m_rectHighlight.height = m_gi->m_rectLabel.height;
                 }
                 else // no text, highlight the icon
                 {
-                    m_gi->m_rectHilight.width = m_gi->m_rectIcon.width;
-                    m_gi->m_rectHilight.height = m_gi->m_rectIcon.height;
+                    m_gi->m_rectHighlight.width = m_gi->m_rectIcon.width;
+                    m_gi->m_rectHighlight.height = m_gi->m_rectIcon.height;
                 }
             }
             break;
@@ -1084,8 +1329,8 @@ void wxListLineData::CalculateSize( wxDC *dc, int spacing )
                         m_gi->m_rectAll.height = h;
                 }
 
-                m_gi->m_rectHilight.width = m_gi->m_rectAll.width;
-                m_gi->m_rectHilight.height = m_gi->m_rectAll.height;
+                m_gi->m_rectHighlight.width = m_gi->m_rectAll.width;
+                m_gi->m_rectHighlight.height = m_gi->m_rectAll.height;
             }
             break;
 
@@ -1128,13 +1373,13 @@ void wxListLineData::SetPosition( int x, int y,
                 else
                     m_gi->m_rectLabel.x = m_gi->m_rectAll.x + 2 + (spacing/2) - (m_gi->m_rectLabel.width/2);
                 m_gi->m_rectLabel.y = m_gi->m_rectAll.y + m_gi->m_rectAll.height + 2 - m_gi->m_rectLabel.height;
-                m_gi->m_rectHilight.x = m_gi->m_rectLabel.x - 2;
-                m_gi->m_rectHilight.y = m_gi->m_rectLabel.y - 2;
+                m_gi->m_rectHighlight.x = m_gi->m_rectLabel.x - 2;
+                m_gi->m_rectHighlight.y = m_gi->m_rectLabel.y - 2;
             }
             else // no text, highlight the icon
             {
-                m_gi->m_rectHilight.x = m_gi->m_rectIcon.x - 4;
-                m_gi->m_rectHilight.y = m_gi->m_rectIcon.y - 4;
+                m_gi->m_rectHighlight.x = m_gi->m_rectIcon.x - 4;
+                m_gi->m_rectHighlight.y = m_gi->m_rectIcon.y - 4;
             }
             break;
 
@@ -1142,8 +1387,8 @@ void wxListLineData::SetPosition( int x, int y,
             m_gi->m_rectAll.x = x;
             m_gi->m_rectAll.y = y;
 
-            m_gi->m_rectHilight.x = m_gi->m_rectAll.x;
-            m_gi->m_rectHilight.y = m_gi->m_rectAll.y;
+            m_gi->m_rectHighlight.x = m_gi->m_rectAll.x;
+            m_gi->m_rectHighlight.y = m_gi->m_rectAll.y;
             m_gi->m_rectLabel.y = m_gi->m_rectAll.y + 2;
 
             if (item->HasImage())
@@ -1167,25 +1412,6 @@ void wxListLineData::SetPosition( int x, int y,
     }
 }
 
-long wxListLineData::IsHit( int x, int y )
-{
-    wxListItemDataList::Node *node = m_items.GetFirst();
-    wxCHECK_MSG( node, 0, _T("no subitems at all??") );
-
-    wxListItemData *item = node->GetData();
-    if ( item->HasImage() && GetIconRect().Inside(x, y) )
-        return wxLIST_HITTEST_ONITEMICON;
-
-    if ( item->HasText() )
-    {
-        wxRect rect = InReportView() ? GetRect() : GetLabelRect();
-        if ( rect.Inside(x, y) )
-            return wxLIST_HITTEST_ONITEMLABEL;
-    }
-
-    return 0;
-}
-
 void wxListLineData::InitItems( int num )
 {
     for (int i = 0; i < num; i++)
@@ -1253,16 +1479,34 @@ int wxListLineData::GetImage( int index ) const
     return item->GetImage();
 }
 
+wxListItemAttr *wxListLineData::GetAttr() const
+{
+    wxListItemDataList::Node *node = m_items.GetFirst();
+    wxCHECK_MSG( node, NULL, _T("invalid column index in GetAttr()") );
+
+    wxListItemData *item = node->GetData();
+    return item->GetAttr();
+}
+
+void wxListLineData::SetAttr(wxListItemAttr *attr)
+{
+    wxListItemDataList::Node *node = m_items.GetFirst();
+    wxCHECK_RET( node, _T("invalid column index in SetAttr()") );
+
+    wxListItemData *item = node->GetData();
+    item->SetAttr(attr);
+}
+
 void wxListLineData::SetAttributes(wxDC *dc,
                                    const wxListItemAttr *attr,
                                    const wxColour& colText,
                                    const wxFont& font,
-                                   bool hilight)
+                                   bool highlight)
 {
     // don't use foregroud colour for drawing highlighted items - this might
     // make them completely invisible (and there is no way to do bit
     // arithmetics on wxColour, unfortunately)
-    if ( !hilight && attr && attr->HasTextColour() )
+    if ( !highlight && attr && attr->HasTextColour() )
     {
         dc->SetTextForeground(attr->GetTextColour());
     }
@@ -1281,23 +1525,39 @@ void wxListLineData::SetAttributes(wxDC *dc,
     }
 }
 
-void wxListLineData::Draw( wxDC *dc, int y, int height, bool hilighted )
+void wxListLineData::Draw( wxDC *dc )
 {
-    wxRect rect = GetRect();
-    m_owner->CalcScrolledPosition( rect.x, rect.y, &rect.x, &rect.y );
+    wxListItemDataList::Node *node = m_items.GetFirst();
+    wxCHECK_RET( node, _T("no subitems at all??") );
 
-    if ( !m_owner->IsExposed( rect ) )
-        return;
+    wxListItemData *item = node->GetData();
+    if (item->HasImage())
+    {
+        wxRect rectIcon = m_gi->m_rectIcon;
+        m_owner->DrawImage( item->GetImage(), dc,
+                            rectIcon.x, rectIcon.y );
+    }
 
-    wxWindow *listctrl = m_owner->GetParent();
+    if (item->HasText())
+    {
+        wxRect rectLabel = m_gi->m_rectLabel;
+        dc->DrawText( item->GetText(), rectLabel.x, rectLabel.y );
+    }
+}
 
+void wxListLineData::DrawInReportMode( wxDC *dc,
+                                       const wxRect& rect,
+                                       const wxRect& rectHL,
+                                       bool highlighted )
+{
     // use our own flag if we maintain it
-    if ( !m_owner->IsVirtual() )
-        hilighted = m_hilighted;
+    if ( !IsVirtual() )
+        highlighted = m_highlighted;
 
     // default foreground colour
+    wxWindow *listctrl = m_owner->GetParent();
     wxColour colText;
-    if ( hilighted )
+    if ( highlighted )
     {
         colText = wxSystemSettings::GetSystemColour( wxSYS_COLOUR_HIGHLIGHTTEXT );
     }
@@ -1309,19 +1569,18 @@ void wxListLineData::Draw( wxDC *dc, int y, int height, bool hilighted )
     // default font
     wxFont font = listctrl->GetFont();
 
-    // VZ: currently we set the colours/fonts only once, but like this (i.e.
-    //     using SetAttributes() inside the loop), it will be trivial to
-    //     customize the subitems (in report mode) too.
-    wxListItemData *item = m_items.GetFirst()->GetData();
-    wxListItemAttr *attr = item->GetAttributes();
-    SetAttributes(dc, attr, colText, font, hilighted);
+    // TODO: later we should support setting different attributes for
+    //       different columns - to do it, just add "col" argument to
+    //       GetAttr() and move this code into the loop below
+    wxListItemAttr *attr = GetAttr();
+    SetAttributes(dc, attr, colText, font, highlighted);
 
     bool hasBgCol = attr && attr->HasBackgroundColour();
-    if ( hilighted || hasBgCol )
+    if ( highlighted || hasBgCol )
     {
-        if ( hilighted )
+        if ( highlighted )
         {
-            dc->SetBrush( *m_owner->m_hilightBrush );
+            dc->SetBrush( *m_owner->m_highlightBrush );
         }
         else
         {
@@ -1332,84 +1591,60 @@ void wxListLineData::Draw( wxDC *dc, int y, int height, bool hilighted )
         }
 
         dc->SetPen( * wxTRANSPARENT_PEN );
-        dc->DrawRectangle( GetHighlightRect() );
+        dc->DrawRectangle( rectHL );
     }
 
     wxListItemDataList::Node *node = m_items.GetFirst();
+    wxCHECK_RET( node, _T("no subitems at all??") );
+
+    size_t col = 0;
+    wxCoord x = rect.x + HEADER_OFFSET_X,
+            y = rect.y + (LINE_SPACING + EXTRA_HEIGHT) / 2;
 
-    if ( GetMode() == wxLC_REPORT)
+    while ( node )
     {
-        size_t col = 0;
-        int x = HEADER_OFFSET_X;
+        wxListItemData *item = node->GetData();
 
-        y += EXTRA_HEIGHT / 2;
+        int xOld = x;
 
-        while ( node )
+        if ( item->HasImage() )
         {
-            wxListItemData *item = node->GetData();
-
-            int xOld = x;
-
-            if ( item->HasImage() )
-            {
-                int ix, iy;
-                m_owner->DrawImage( item->GetImage(), dc, x, y );
-                m_owner->GetImageSize( item->GetImage(), ix, iy );
-                x += ix + 5; // FIXME: what is "5"?
-            }
-
-            int width = m_owner->GetColumnWidth(col++);
-
-            dc->SetClippingRegion(x, y, width, height);
-
-            if ( item->HasText() )
-            {
-                dc->DrawText( item->GetText(), x, y + 1 );
-            }
+            int ix, iy;
+            m_owner->DrawImage( item->GetImage(), dc, x, y );
+            m_owner->GetImageSize( item->GetImage(), ix, iy );
+            x += ix + 5; // FIXME: what is "5"?
+        }
 
-            dc->DestroyClippingRegion();
+        int width = m_owner->GetColumnWidth(col++);
 
-            x = xOld + width;
+        wxDCClipper clipper(*dc, x, y, width, rect.height);
 
-            node = node->GetNext();
-        }
-    }
-    else // !report
-    {
-        if (node)
+        if ( item->HasText() )
         {
-            wxListItemData *item = node->GetData();
-            if (item->HasImage())
-            {
-                wxRect rectIcon = GetIconRect();
-                m_owner->DrawImage( item->GetImage(), dc,
-                                    rectIcon.x, rectIcon.y );
-            }
-
-            if (item->HasText())
-            {
-                wxRect rectLabel = GetLabelRect();
-                dc->DrawText( item->GetText(), rectLabel.x, rectLabel.y );
-            }
+            dc->DrawText( item->GetText(), x, y );
         }
+
+        x = xOld + width;
+
+        node = node->GetNext();
     }
 }
 
-bool wxListLineData::Hilight( bool on )
+bool wxListLineData::Highlight( bool on )
 {
-    wxCHECK_MSG( !m_owner->IsVirtual(), FALSE, _T("unexpected call to Hilight") );
+    wxCHECK_MSG( !m_owner->IsVirtual(), FALSE, _T("unexpected call to Highlight") );
 
-    if ( on == m_hilighted )
+    if ( on == m_highlighted )
         return FALSE;
 
-    m_hilighted = on;
+    m_highlighted = on;
 
     return TRUE;
 }
 
-void wxListLineData::ReverseHilight( void )
+void wxListLineData::ReverseHighlight( void )
 {
-    Hilight(!IsHilighted());
+    Highlight(!IsHighlighted());
 }
 
 //-----------------------------------------------------------------------------
@@ -1455,13 +1690,15 @@ wxListHeaderWindow::~wxListHeaderWindow( void )
 void wxListHeaderWindow::DoDrawRect( wxDC *dc, int x, int y, int w, int h )
 {
 #ifdef __WXGTK__
-    GtkStateType state = GTK_STATE_NORMAL;
-    if (!m_parent->IsEnabled()) state = GTK_STATE_INSENSITIVE;
+    GtkStateType state = m_parent->IsEnabled() ? GTK_STATE_NORMAL
+                                               : GTK_STATE_INSENSITIVE;
 
     x = dc->XLOG2DEV( x );
 
-    gtk_paint_box (m_wxwindow->style, GTK_PIZZA(m_wxwindow)->bin_window, state, GTK_SHADOW_OUT,
-                   (GdkRectangle*) NULL, m_wxwindow, "button", x-1, y-1, w+2, h+2);
+    gtk_paint_box (m_wxwindow->style, GTK_PIZZA(m_wxwindow)->bin_window,
+                   state, GTK_SHADOW_OUT,
+                   (GdkRectangle*) NULL, m_wxwindow, "button",
+                   x-1, y-1, w+2, h+2);
 #elif defined( __WXMAC__  )
     const int m_corner = 1;
 
@@ -1482,7 +1719,7 @@ void wxListHeaderWindow::DoDrawRect( wxDC *dc, int x, int y, int w, int h )
     dc->DrawRectangle( x, y, 1, h );              // left (outer)
     dc->DrawLine( x, y+h-1, x+1, y+h-1 );
     dc->DrawLine( x+w-1, y, x+w-1, y+1 );
-#else
+#else // !GTK, !Mac
     const int m_corner = 1;
 
     dc->SetBrush( *wxTRANSPARENT_BRUSH );
@@ -1822,7 +2059,6 @@ IMPLEMENT_DYNAMIC_CLASS(wxListMainWindow,wxScrolledWindow);
 
 BEGIN_EVENT_TABLE(wxListMainWindow,wxScrolledWindow)
   EVT_PAINT          (wxListMainWindow::OnPaint)
-  EVT_SIZE           (wxListMainWindow::OnSize)
   EVT_MOUSE_EVENTS   (wxListMainWindow::OnMouse)
   EVT_CHAR           (wxListMainWindow::OnChar)
   EVT_KEY_DOWN       (wxListMainWindow::OnKeyDown)
@@ -1837,7 +2073,7 @@ void wxListMainWindow::Init()
     m_dirty = TRUE;
     m_countVirt = 0;
     m_lineFrom =
-    m_lineTo = 0;
+    m_lineTo = (size_t)-1;
     m_linesPerPage = 0;
 
     m_headerWidth =
@@ -1881,7 +2117,7 @@ wxListMainWindow::wxListMainWindow()
 {
     Init();
 
-    m_hilightBrush = (wxBrush *) NULL;
+    m_highlightBrush = (wxBrush *) NULL;
 
     m_xScroll =
     m_yScroll = 0;
@@ -1898,7 +2134,7 @@ wxListMainWindow::wxListMainWindow( wxWindow *parent,
 {
     Init();
 
-    m_hilightBrush = new wxBrush( wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT), wxSOLID );
+    m_highlightBrush = new wxBrush( wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT), wxSOLID );
     wxSize sz = size;
     sz.y = 25;
 
@@ -1912,7 +2148,7 @@ wxListMainWindow::~wxListMainWindow()
 {
     DeleteEverything();
 
-    delete m_hilightBrush;
+    delete m_highlightBrush;
 
     delete m_renameTimer;
 }
@@ -1921,7 +2157,7 @@ void wxListMainWindow::CacheLineData(size_t line)
 {
     wxListCtrl *listctrl = GetListCtrl();
 
-    wxListLineData *ld = GetFirstLine();
+    wxListLineData *ld = GetDummyLine();
 
     size_t countCol = GetColumnCount();
     for ( size_t col = 0; col < countCol; col++ )
@@ -1929,11 +2165,11 @@ void wxListMainWindow::CacheLineData(size_t line)
         ld->SetText(col, listctrl->OnGetItemText(line, col));
     }
 
-    ld->SetImage(0, listctrl->OnGetItemImage(line));
-    ld->SetLineIndex(line);
+    ld->SetImage(listctrl->OnGetItemImage(line));
+    ld->SetAttr(listctrl->OnGetItemAttr(line));
 }
 
-wxListLineData *wxListMainWindow::GetFirstLine() const
+wxListLineData *wxListMainWindow::GetDummyLine() const
 {
     wxASSERT_MSG( !IsEmpty(), _T("invalid line index") );
 
@@ -1943,101 +2179,180 @@ wxListLineData *wxListMainWindow::GetFirstLine() const
         // already if it's not empty
         wxASSERT_MSG( IsVirtual(), _T("logic error") );
 
-        wxListMainWindow *self = wxConstCast(this, wxListMainWindow);
-        wxListLineData *line = new wxListLineData( self, 0 );
-        self->m_lines.Add(line);
-    }
+        wxListMainWindow *self = wxConstCast(this, wxListMainWindow);
+        wxListLineData *line = new wxListLineData(self);
+        self->m_lines.Add(line);
+    }
+
+    return &m_lines[0];
+}
+
+// ----------------------------------------------------------------------------
+// line geometry (report mode only)
+// ----------------------------------------------------------------------------
+
+wxCoord wxListMainWindow::GetLineHeight() const
+{
+    wxASSERT_MSG( HasFlag(wxLC_REPORT), _T("only works in report mode") );
+
+    // we cache the line height as calling GetTextExtent() is slow
+    if ( !m_lineHeight )
+    {
+        wxListMainWindow *self = wxConstCast(this, wxListMainWindow);
+
+        wxClientDC dc( self );
+        dc.SetFont( GetFont() );
+
+        wxCoord y;
+        dc.GetTextExtent(_T("H"), NULL, &y);
+
+        if ( y < SCROLL_UNIT_Y )
+            y = SCROLL_UNIT_Y;
+        y += EXTRA_HEIGHT;
+
+        self->m_lineHeight = y + LINE_SPACING;
+    }
+
+    return m_lineHeight;
+}
+
+wxCoord wxListMainWindow::GetLineY(size_t line) const
+{
+    wxASSERT_MSG( HasFlag(wxLC_REPORT), _T("only works in report mode") );
+
+    return LINE_SPACING + line*GetLineHeight();
+}
+
+wxRect wxListMainWindow::GetLineRect(size_t line) const
+{
+    if ( !InReportView() )
+        return GetLine(line)->m_gi->m_rectAll;
+
+    wxRect rect;
+    rect.x = HEADER_OFFSET_X;
+    rect.y = GetLineY(line);
+    rect.width = GetHeaderWidth();
+    rect.height = GetLineHeight();
+
+    return rect;
+}
+
+wxRect wxListMainWindow::GetLineLabelRect(size_t line) const
+{
+    if ( !InReportView() )
+        return GetLine(line)->m_gi->m_rectLabel;
+
+    wxRect rect;
+    rect.x = HEADER_OFFSET_X;
+    rect.y = GetLineY(line);
+    rect.width = GetColumnWidth(0);
+    rect.height = GetLineHeight();
+
+    return rect;
+}
+
+wxRect wxListMainWindow::GetLineIconRect(size_t line) const
+{
+    if ( !InReportView() )
+        return GetLine(line)->m_gi->m_rectIcon;
+
+    wxListLineData *ld = GetLine(line);
+    wxASSERT_MSG( ld->HasImage(), _T("should have an image") );
 
-    m_lines[0].SetLineIndex(0);
+    wxRect rect;
+    rect.x = HEADER_OFFSET_X;
+    rect.y = GetLineY(line);
+    GetImageSize(ld->GetImage(), rect.width, rect.height);
 
-    return &m_lines[0];
+    return rect;
 }
 
-wxCoord wxListMainWindow::GetLineHeight() const
+wxRect wxListMainWindow::GetLineHighlightRect(size_t line) const
 {
-    wxASSERT_MSG( HasFlag(wxLC_REPORT), _T("only works in report mode") );
-
-    // we cache the line height as calling GetTextExtent() is slow
-    if ( !m_lineHeight )
-    {
-        wxListMainWindow *self = wxConstCast(this, wxListMainWindow);
+    return InReportView() ? GetLineRect(line)
+                          : GetLine(line)->m_gi->m_rectHighlight;
+}
 
-        wxClientDC dc( self );
-        dc.SetFont( GetFont() );
+long wxListMainWindow::HitTestLine(size_t line, int x, int y) const
+{
+    wxListLineData *ld = GetLine(line);
 
-        wxCoord y;
-        dc.GetTextExtent(_T("H"), NULL, &y);
+    if ( ld->HasImage() && GetLineIconRect(line).Inside(x, y) )
+        return wxLIST_HITTEST_ONITEMICON;
 
-        if ( y < SCROLL_UNIT_Y )
-            y = SCROLL_UNIT_Y;
-        y += EXTRA_HEIGHT;
+    if ( ld->HasText() )
+    {
+        wxRect rect = InReportView() ? GetLineRect(line)
+                                     : GetLineLabelRect(line);
 
-        self->m_lineHeight = y + LINE_SPACING;
+        if ( rect.Inside(x, y) )
+            return wxLIST_HITTEST_ONITEMLABEL;
     }
 
-    return m_lineHeight;
+    return 0;
 }
 
-wxCoord wxListMainWindow::GetLineY(size_t line) const
+// ----------------------------------------------------------------------------
+// highlight (selection) handling
+// ----------------------------------------------------------------------------
+
+bool wxListMainWindow::IsHighlighted(size_t line) const
 {
-    wxASSERT_MSG( HasFlag(wxLC_REPORT), _T("only works in report mode") );
+    if ( IsVirtual() )
+    {
+        return m_selStore.IsSelected(line);
+    }
+    else // !virtual
+    {
+        wxListLineData *ld = GetLine(line);
+        wxCHECK_MSG( ld, FALSE, _T("invalid index in IsHighlighted") );
 
-    return LINE_SPACING + line*GetLineHeight();
+        return ld->IsHighlighted();
+    }
 }
 
-bool wxListMainWindow::IsHilighted(size_t line) const
+void wxListMainWindow::HighlightLines( size_t lineFrom, size_t lineTo, bool highlight )
 {
     if ( IsVirtual() )
     {
-        return m_selections.Index(line) != wxNOT_FOUND;
+        m_selStore.SelectRange(lineFrom, lineTo, highlight);
+        RefreshLines(lineFrom, lineTo);
     }
-    else // !virtual
+    else
     {
-        wxListLineData *ld = GetLine(line);
-        wxCHECK_MSG( ld, FALSE, _T("invalid index in IsHilighted") );
+        // do it the dumb way
+        bool needsRefresh = FALSE;
+        for ( size_t line = lineFrom; line <= lineTo; line++ )
+        {
+            if ( HighlightLine(line, highlight) )
+                needsRefresh = TRUE;
+        }
 
-        return ld->IsHilighted();
+        if ( needsRefresh )
+            RefreshLines(lineFrom, lineTo);
     }
 }
 
-bool wxListMainWindow::HilightLine( size_t line, bool hilight )
+bool wxListMainWindow::HighlightLine( size_t line, bool highlight )
 {
     bool changed;
 
     if ( IsVirtual() )
     {
-        changed = FALSE;
-
-        int index = m_selections.Index(line);
-        if ( hilight )
-        {
-            if ( index == wxNOT_FOUND )
-            {
-                m_selections.Add(line);
-                changed = TRUE;
-            }
-        }
-        else // !hilight
-        {
-            if ( index != wxNOT_FOUND )
-            {
-                m_selections.RemoveAt((size_t)index);
-                changed = TRUE;
-            }
-        }
+        changed = m_selStore.SelectItem(line, highlight);
     }
     else // !virtual
     {
         wxListLineData *ld = GetLine(line);
-        wxCHECK_MSG( ld, FALSE, _T("invalid index in IsHilighted") );
+        wxCHECK_MSG( ld, FALSE, _T("invalid index in IsHighlighted") );
 
-        changed = ld->Hilight(hilight);
+        changed = ld->Highlight(highlight);
     }
 
     if ( changed )
     {
-        SendNotify( line, hilight ? wxEVT_COMMAND_LIST_ITEM_SELECTED
-                                  : wxEVT_COMMAND_LIST_ITEM_DESELECTED );
+        SendNotify( line, highlight ? wxEVT_COMMAND_LIST_ITEM_SELECTED
+                                    : wxEVT_COMMAND_LIST_ITEM_DESELECTED );
     }
 
     return changed;
@@ -2045,7 +2360,7 @@ bool wxListMainWindow::HilightLine( size_t line, bool hilight )
 
 void wxListMainWindow::RefreshLine( size_t line )
 {
-    wxRect rect = GetLine(line)->GetRect();
+    wxRect rect = GetLineRect(line);
 
     CalcScrolledPosition( rect.x, rect.y, &rect.x, &rect.y );
     RefreshRect( rect );
@@ -2056,12 +2371,17 @@ void wxListMainWindow::RefreshLines( size_t lineFrom, size_t lineTo )
     // we suppose that they are ordered by caller
     wxASSERT_MSG( lineFrom <= lineTo, _T("indices in disorder") );
 
+    wxASSERT_MSG( lineTo < GetItemCount(), _T("invalid line range") );
+
     if ( HasFlag(wxLC_REPORT) )
     {
-        if ( lineFrom < m_lineFrom )
-            lineFrom = m_lineFrom;
-        if ( lineTo > m_lineTo )
-            lineTo = m_lineTo;
+        size_t visibleFrom, visibleTo;
+        GetVisibleLinesRange(&visibleFrom, &visibleTo);
+
+        if ( lineFrom < visibleFrom )
+            lineFrom = visibleFrom;
+        if ( lineTo > visibleTo )
+            lineTo = visibleTo;
 
         wxRect rect;
         rect.x = 0;
@@ -2082,21 +2402,50 @@ void wxListMainWindow::RefreshLines( size_t lineFrom, size_t lineTo )
     }
 }
 
+void wxListMainWindow::RefreshAfter( size_t lineFrom )
+{
+    if ( HasFlag(wxLC_REPORT) )
+    {
+        size_t visibleFrom;
+        GetVisibleLinesRange(&visibleFrom, NULL);
+
+        if ( lineFrom < visibleFrom )
+            lineFrom = visibleFrom;
+
+        wxRect rect;
+        rect.x = 0;
+        rect.y = GetLineY(lineFrom);
+
+        wxSize size = GetClientSize();
+        rect.width = size.x;
+        // refresh till the bottom of the window
+        rect.height = size.y - rect.y;
+
+        CalcScrolledPosition( rect.x, rect.y, &rect.x, &rect.y );
+        RefreshRect( rect );
+    }
+    else // !report
+    {
+        // TODO: how to do it more efficiently?
+        m_dirty = TRUE;
+    }
+}
+
 void wxListMainWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
 {
     // Note: a wxPaintDC must be constructed even if no drawing is
     // done (a Windows requirement).
     wxPaintDC dc( this );
 
-    if ( m_dirty )
+    if ( IsEmpty() )
     {
-        // postpone redrawing until the next OnIdle() call to minimize flicker
+        // empty control. nothing to draw
         return;
     }
 
-    if ( IsEmpty() )
+    if ( m_dirty )
     {
-        // empty control. nothing to draw
+        // delay the repainting until we calculate all the items positions
         return;
     }
 
@@ -2111,53 +2460,59 @@ void wxListMainWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
 
     if ( HasFlag(wxLC_REPORT) )
     {
-        int lineSpacing = GetLineHeight();
+        int lineHeight = GetLineHeight();
+
+        size_t visibleFrom, visibleTo;
+        GetVisibleLinesRange(&visibleFrom, &visibleTo);
 
-        for ( size_t line = m_lineFrom; line <= m_lineTo; line++ )
+        wxRect rectLine;
+        wxCoord xOrig, yOrig;
+        CalcUnscrolledPosition(0, 0, &xOrig, &yOrig);
+
+        for ( size_t line = visibleFrom; line <= visibleTo; line++ )
         {
-            GetLine(line)->Draw( &dc,
-                                 GetLineY(line),
-                                 lineSpacing,
-                                 IsHilighted(line) );
+            rectLine = GetLineRect(line);
+
+            if ( !IsExposed(rectLine.x - xOrig, rectLine.y - yOrig,
+                            rectLine.width, rectLine.height) )
+            {
+                // don't redraw unaffected lines to avoid flicker
+                continue;
+            }
+
+            GetLine(line)->DrawInReportMode( &dc,
+                                             rectLine,
+                                             GetLineHighlightRect(line),
+                                             IsHighlighted(line) );
         }
 
         if ( HasFlag(wxLC_HRULES) )
         {
-#ifdef __WXMAC__
-            wxPen pen(wxWHITE, 1, wxSOLID);
-#else
-            wxPen pen(wxSystemSettings::
-                        GetSystemColour(wxSYS_COLOUR_3DLIGHT), 1, wxSOLID);
-#endif
+            wxPen pen(GetRuleColour(), 1, wxSOLID);
             wxSize clientSize = GetClientSize();
 
-            for ( size_t i = m_lineFrom; i <= m_lineTo; i++ )
+            for ( size_t i = visibleFrom; i <= visibleTo; i++ )
             {
                 dc.SetPen(pen);
                 dc.SetBrush( *wxTRANSPARENT_BRUSH );
-                dc.DrawLine(0 - dev_x, i*lineSpacing,
-                            clientSize.x - dev_x, i*lineSpacing);
+                dc.DrawLine(0 - dev_x, i*lineHeight,
+                            clientSize.x - dev_x, i*lineHeight);
             }
 
             // Draw last horizontal rule
-            if ( m_lineTo > m_lineFrom )
+            if ( visibleTo > visibleFrom )
             {
                 dc.SetPen(pen);
                 dc.SetBrush( *wxTRANSPARENT_BRUSH );
-                dc.DrawLine(0 - dev_x, m_lineTo*lineSpacing,
-                            clientSize.x - dev_x , m_lineTo*lineSpacing );
+                dc.DrawLine(0 - dev_x, m_lineTo*lineHeight,
+                            clientSize.x - dev_x , m_lineTo*lineHeight );
             }
         }
 
         // Draw vertical rules if required
         if ( HasFlag(wxLC_VRULES) && !IsEmpty() )
         {
-#ifdef __WXMAC__
-            wxPen pen(wxWHITE, 1, wxSOLID);
-#else
-            wxPen pen(wxSystemSettings::
-                        GetSystemColour(wxSYS_COLOUR_3DLIGHT), 1, wxSOLID);
-#endif
+            wxPen pen(GetRuleColour(), 1, wxSOLID);
 
             int col = 0;
             wxRect firstItemRect;
@@ -2187,54 +2542,34 @@ void wxListMainWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
 
     if ( HasCurrent() && m_hasFocus )
     {
-        wxRect rect;
-
-        if ( IsVirtual() )
-        {
-            // just offset the rect of the first line to position it correctly
-            wxListLineData *line = GetFirstLine();
-            rect = line->GetHighlightRect();
-            rect.y = GetLineY(m_current);
-        }
-        else
-        {
-            rect = GetLine(m_current)->GetHighlightRect();
-        }
 #ifdef __WXMAC__
         // no rect outline, we already have the background color
 #else
         dc.SetPen( *wxBLACK_PEN );
         dc.SetBrush( *wxTRANSPARENT_BRUSH );
-        dc.DrawRectangle( rect );
+        dc.DrawRectangle( GetLineHighlightRect(m_current) );
 #endif
     }
 
     dc.EndDrawing();
 }
 
-void wxListMainWindow::HilightAll( bool on )
+void wxListMainWindow::HighlightAll( bool on )
 {
-    bool needsRefresh = FALSE;
-
-    size_t count = GetItemCount();
-    for ( size_t line = 0; line < count; line++ )
+    if ( IsSingleSel() )
     {
-        if ( HilightLine( line, on ) )
+        wxASSERT_MSG( !on, _T("can't do this in a single sel control") );
+
+        // we just have one item to turn off
+        if ( HasCurrent() && IsHighlighted(m_current) )
         {
-            if ( HasFlag(wxLC_REPORT) )
-            {
-                needsRefresh = TRUE;
-            }
-            else
-            {
-                RefreshLine(line);
-            }
+            HighlightLine(m_current, FALSE);
+            RefreshLine(m_current);
         }
     }
-
-    if ( needsRefresh )
+    else // multi sel
     {
-        RefreshLines( 0, count - 1 );
+        HighlightLines(0, GetItemCount() - 1, on);
     }
 }
 
@@ -2291,7 +2626,7 @@ void wxListMainWindow::EditLabel( long item )
     PrepareDC( dc );
 
     wxString s = data->GetText(0);
-    wxRect rectLabel = data->GetLabelRect();
+    wxRect rectLabel = GetLineLabelRect(m_currentEdit);
 
     rectLabel.x = dc.LogicalToDeviceX( rectLabel.x );
     rectLabel.y = dc.LogicalToDeviceY( rectLabel.y );
@@ -2359,7 +2694,7 @@ void wxListMainWindow::OnMouse( wxMouseEvent &event )
     int y = event.GetY();
     CalcUnscrolledPosition( x, y, &x, &y );
 
-    /* Did we actually hit an item ? */
+    // where did we hit it (if we did)?
     long hitResult = 0;
 
     size_t count = GetItemCount(),
@@ -2367,10 +2702,9 @@ void wxListMainWindow::OnMouse( wxMouseEvent &event )
 
     if ( HasFlag(wxLC_REPORT) )
     {
-        wxCoord lineHeight = GetLineHeight();
-
-        current = y / lineHeight;
-        hitResult = GetFirstLine()->IsHit( x, y % lineHeight );
+        current = y / GetLineHeight();
+        if ( current < count )
+            hitResult = HitTestLine(current, x, y);
     }
     else // !report
     {
@@ -2378,10 +2712,8 @@ void wxListMainWindow::OnMouse( wxMouseEvent &event )
         //       enumerating all items is still not a way to do it!!
         for ( current = 0; current < count; current++ )
         {
-            wxListLineData *line = (wxListLineData *) NULL;
-            line = GetLine(current);
-            hitResult = line->IsHit( x, y );
-            if (hitResult)
+            hitResult = HitTestLine(current, x, y);
+            if ( hitResult )
                 break;
         }
     }
@@ -2463,21 +2795,20 @@ void wxListMainWindow::OnMouse( wxMouseEvent &event )
 
         size_t oldCurrent = m_current;
 
-        if ( HasFlag(wxLC_SINGLE_SEL) ||
-             !(event.ControlDown() || event.ShiftDown()) )
+        if ( IsSingleSel() || !(event.ControlDown() || event.ShiftDown()) )
         {
+            HighlightAll( FALSE );
             m_current = current;
-            HilightAll( FALSE );
 
-            ReverseHilight(m_current);
+            ReverseHighlight(m_current);
         }
-        else // multi sel
+        else // multi sel & either ctrl or shift is down
         {
             if (event.ControlDown())
             {
                 m_current = current;
 
-                ReverseHilight(m_current);
+                ReverseHighlight(m_current);
             }
             else if (event.ShiftDown())
             {
@@ -2492,17 +2823,7 @@ void wxListMainWindow::OnMouse( wxMouseEvent &event )
                     lineFrom = m_current;
                 }
 
-                bool needsRefresh = FALSE;
-                for ( size_t i = lineFrom; i <= lineTo; i++ )
-                {
-                    if ( HilightLine(i, TRUE) )
-                        needsRefresh = TRUE;
-                }
-
-                if ( needsRefresh )
-                {
-                    RefreshLines(lineFrom, lineTo);
-                }
+                HighlightLines(lineFrom, lineTo);
             }
             else // !ctrl, !shift
             {
@@ -2525,11 +2846,10 @@ void wxListMainWindow::OnMouse( wxMouseEvent &event )
 
 void wxListMainWindow::MoveToFocus()
 {
-    if (!HasCurrent())
+    if ( !HasCurrent() )
         return;
 
-    wxListLineData *data = IsVirtual() ? GetFirstLine() : GetLine(m_current);
-    wxRect rect = data->GetRect();
+    wxRect rect = GetLineRect(m_current);
 
     int client_w, client_h;
     GetClientSize( &client_w, &client_h );
@@ -2539,14 +2859,16 @@ void wxListMainWindow::MoveToFocus()
 
     if ( HasFlag(wxLC_REPORT) )
     {
+        // the next we need the range of lines shown it might be different, so
+        // recalculate it
+        ResetVisibleLinesRange();
+
         if (rect.y < view_y )
             Scroll( -1, rect.y/m_yScroll );
         if (rect.y+rect.height+5 > view_y+client_h)
             Scroll( -1, (rect.y+rect.height-client_h+SCROLL_UNIT_Y)/m_yScroll );
-
-        UpdateShownLinesRange();
     }
-    else
+    else // !report
     {
         if (rect.x-view_x < 5)
             Scroll( (rect.x-5)/m_xScroll, -1 );
@@ -2568,7 +2890,7 @@ void wxListMainWindow::OnArrowChar(size_t newCurrent, const wxKeyEvent& event)
 
     // in single selection we just ignore Shift as we can't select several
     // items anyhow
-    if ( event.ShiftDown() && !HasFlag(wxLC_SINGLE_SEL) )
+    if ( event.ShiftDown() && !IsSingleSel() )
     {
         m_current = newCurrent;
 
@@ -2579,28 +2901,22 @@ void wxListMainWindow::OnArrowChar(size_t newCurrent, const wxKeyEvent& event)
             oldCurrent = m_current;
         }
 
-        bool needsRefresh = FALSE;
-        for ( size_t line = oldCurrent; line <= newCurrent; line++ )
-        {
-            if ( HilightLine( line ) )
-            {
-                needsRefresh = TRUE;
-            }
-        }
-
-        if ( needsRefresh )
-            RefreshLines( oldCurrent, newCurrent );
+        HighlightLines(oldCurrent, newCurrent);
     }
     else // !shift
     {
+        // all previously selected items are unselected unless ctrl is held
+        if ( !event.ControlDown() )
+            HighlightAll(FALSE);
+
         m_current = newCurrent;
 
-        HilightLine( oldCurrent, FALSE );
+        HighlightLine( oldCurrent, FALSE );
         RefreshLine( oldCurrent );
 
         if ( !event.ControlDown() )
         {
-            HilightLine( m_current, TRUE );
+            HighlightLine( m_current, TRUE );
         }
     }
 
@@ -2762,7 +3078,7 @@ void wxListMainWindow::OnChar( wxKeyEvent &event )
             break;
 
         case WXK_SPACE:
-            if ( HasFlag(wxLC_SINGLE_SEL) )
+            if ( IsSingleSel() )
             {
                 wxListEvent le( wxEVT_COMMAND_LIST_ITEM_ACTIVATED,
                                 GetParent()->GetId() );
@@ -2773,7 +3089,7 @@ void wxListMainWindow::OnChar( wxKeyEvent &event )
             }
             else
             {
-                ReverseHilight(m_current);
+                ReverseHighlight(m_current);
             }
             break;
 
@@ -2849,7 +3165,7 @@ void wxListMainWindow::DrawImage( int index, wxDC *dc, int x, int y )
     }
 }
 
-void wxListMainWindow::GetImageSize( int index, int &width, int &height )
+void wxListMainWindow::GetImageSize( int index, int &width, int &height ) const
 {
     if ( HasFlag(wxLC_ICON) && m_normal_image_list )
     {
@@ -2874,9 +3190,9 @@ void wxListMainWindow::GetImageSize( int index, int &width, int &height )
     }
 }
 
-int wxListMainWindow::GetTextLength( const wxString &s )
+int wxListMainWindow::GetTextLength( const wxString &s ) const
 {
-    wxClientDC dc( this );
+    wxClientDC dc( wxConstCast(this, wxListMainWindow) );
     dc.SetFont( GetFont() );
 
     wxCoord lw;
@@ -3071,19 +3387,21 @@ void wxListMainWindow::SetItem( wxListItem &item )
     wxCHECK_RET( id >= 0 && (size_t)id < GetItemCount(),
                  _T("invalid item index in SetItem") );
 
-    if ( IsVirtual() )
+    if ( !IsVirtual() )
+    {
+        wxListLineData *line = GetLine((size_t)id);
+        line->SetItem( item.m_col, item );
+    }
+
+    if ( InReportView() )
     {
         // just refresh the line to show the new value of the text/image
         RefreshLine((size_t)id);
     }
-    else // !virtual
+    else // !report
     {
+        // refresh everything (resulting in horrible flicker - FIXME!)
         m_dirty = TRUE;
-
-        wxListLineData *line = GetLine((size_t)id);
-        if ( HasFlag(wxLC_REPORT) )
-            item.m_width = GetColumnWidth( item.m_col );
-        line->SetItem( item.m_col, item );
     }
 }
 
@@ -3106,9 +3424,9 @@ void wxListMainWindow::SetItemState( long litem, long state, long stateMask )
                 m_current = item;
                 OnFocusLine( m_current );
 
-                if ( HasFlag(wxLC_SINGLE_SEL) && (oldCurrent != (size_t)-1) )
+                if ( IsSingleSel() && (oldCurrent != (size_t)-1) )
                 {
-                    HilightLine(oldCurrent, FALSE);
+                    HighlightLine(oldCurrent, FALSE);
                     RefreshLine(oldCurrent);
                 }
 
@@ -3130,7 +3448,7 @@ void wxListMainWindow::SetItemState( long litem, long state, long stateMask )
     {
         bool on = (state & wxLIST_STATE_SELECTED) != 0;
 
-        if ( HasFlag(wxLC_SINGLE_SEL) )
+        if ( IsSingleSel() )
         {
             if ( on )
             {
@@ -3144,7 +3462,7 @@ void wxListMainWindow::SetItemState( long litem, long state, long stateMask )
 
                     if ( oldCurrent != (size_t)-1 )
                     {
-                        HilightLine( oldCurrent, FALSE );
+                        HighlightLine( oldCurrent, FALSE );
                         RefreshLine( oldCurrent );
                     }
                 }
@@ -3157,7 +3475,7 @@ void wxListMainWindow::SetItemState( long litem, long state, long stateMask )
             }
         }
 
-        if ( HilightLine(item, on) )
+        if ( HighlightLine(item, on) )
         {
             RefreshLine(item);
         }
@@ -3179,7 +3497,7 @@ int wxListMainWindow::GetItemState( long item, long stateMask )
 
     if ( stateMask & wxLIST_STATE_SELECTED )
     {
-        if ( IsHilighted(item) )
+        if ( IsHighlighted(item) )
             ret |= wxLIST_STATE_SELECTED;
     }
 
@@ -3206,6 +3524,7 @@ size_t wxListMainWindow::GetItemCount() const
 
 void wxListMainWindow::SetItemCount(long count)
 {
+    m_selStore.SetItemCount(count);
     m_countVirt = count;
 
     Refresh();
@@ -3214,14 +3533,14 @@ void wxListMainWindow::SetItemCount(long count)
 int wxListMainWindow::GetSelectedItemCount()
 {
     // deal with the quick case first
-    if ( HasFlag(wxLC_SINGLE_SEL) )
+    if ( IsSingleSel() )
     {
-        return m_current == (size_t)-1 ? FALSE : IsHilighted(m_current);
+        return HasCurrent() ? IsHighlighted(m_current) : FALSE;
     }
 
     // virtual controls remmebers all its selections itself
     if ( IsVirtual() )
-        return m_selections.GetCount();
+        return m_selStore.GetSelectedCount();
 
     // TODO: we probably should maintain the number of items selected even for
     //       non virtual controls as enumerating all lines is really slow...
@@ -3229,7 +3548,7 @@ int wxListMainWindow::GetSelectedItemCount()
     size_t count = GetItemCount();
     for ( size_t line = 0; line < count; line++ )
     {
-        if ( GetLine(line)->IsHilighted() )
+        if ( GetLine(line)->IsHighlighted() )
             countSel++;
     }
 
@@ -3245,7 +3564,8 @@ void wxListMainWindow::GetItemRect( long index, wxRect &rect )
     wxCHECK_RET( index >= 0 && (size_t)index < GetItemCount(),
                  _T("invalid index in GetItemRect") );
 
-    rect = GetLine((size_t)index)->GetRect();
+    rect = GetLineRect((size_t)index);
+
     CalcScrolledPosition(rect.x, rect.y, &rect.x, &rect.y);
 }
 
@@ -3264,13 +3584,7 @@ bool wxListMainWindow::GetItemPosition(long item, wxPoint& pos)
 // geometry calculation
 // ----------------------------------------------------------------------------
 
-void wxListMainWindow::OnSize( wxSizeEvent &WXUNUSED(event) )
-{
-    // wait for the next OnIdle() with geometry recalculation
-    m_dirty = TRUE;
-}
-
-void wxListMainWindow::CalculatePositions()
+void wxListMainWindow::RecalculatePositions()
 {
     if ( IsEmpty() )
         return;
@@ -3293,15 +3607,17 @@ void wxListMainWindow::CalculatePositions()
     if ( HasFlag(wxLC_REPORT) )
     {
         // all lines have the same height
-        int lineSpacing = GetLineHeight();
+        int lineHeight = GetLineHeight();
 
         // scroll one line per step
-        m_yScroll = lineSpacing;
+        m_yScroll = lineHeight;
 
         size_t lineCount = GetItemCount();
-        int entireHeight = lineCount*lineSpacing + 2*LINE_SPACING;
+        int entireHeight = lineCount*lineHeight + LINE_SPACING;
 
-        m_linesPerPage = clientHeight / lineSpacing;
+        m_linesPerPage = clientHeight / lineHeight;
+
+        ResetVisibleLinesRange();
 
         SetScrollbars( m_xScroll, m_yScroll,
                        (GetHeaderWidth() + m_xScroll - 1)/m_xScroll,
@@ -3309,8 +3625,6 @@ void wxListMainWindow::CalculatePositions()
                        GetScrollPos(wxHORIZONTAL),
                        GetScrollPos(wxVERTICAL),
                        TRUE );
-
-        UpdateShownLinesRange();
     }
     else // !report
     {
@@ -3320,7 +3634,7 @@ void wxListMainWindow::CalculatePositions()
 
         clientHeight -= 4;  // sunken frame
 
-        int entireWidth;
+        int entireWidth = 0;
 
         for (int tries = 0; tries < 2; tries++)
         {
@@ -3339,7 +3653,7 @@ void wxListMainWindow::CalculatePositions()
                 line->CalculateSize( &dc, iconSpacing );
                 line->SetPosition( x, y, clientWidth, iconSpacing );
 
-                wxSize sizeLine = line->GetSize();
+                wxSize sizeLine = GetLineSize(i);
 
                 if ( maxWidth < sizeLine.x )
                     maxWidth = sizeLine.x;
@@ -3377,11 +3691,26 @@ void wxListMainWindow::CalculatePositions()
 
     // FIXME: why should we call it from here?
     UpdateCurrent();
+
+    RefreshAll();
+}
+
+void wxListMainWindow::RefreshAll()
+{
+    m_dirty = FALSE;
+    Refresh();
+
+    wxListHeaderWindow *headerWin = GetListCtrl()->m_headerWin;
+    if ( headerWin )
+    {
+        headerWin->m_dirty = FALSE;
+        headerWin->Refresh();
+    }
 }
 
 void wxListMainWindow::UpdateCurrent()
 {
-    if ( (m_current == (size_t)-1) && !IsEmpty() )
+    if ( !HasCurrent() && !IsEmpty() )
     {
         m_current = 0;
     }
@@ -3424,7 +3753,7 @@ long wxListMainWindow::GetNextItem( long item,
         if ( (state & wxLIST_STATE_FOCUSED) && (line == m_current) )
             return line;
 
-        if ( (state & wxLIST_STATE_SELECTED) && IsHilighted(line) )
+        if ( (state & wxLIST_STATE_SELECTED) && IsHighlighted(line) )
             return line;
     }
 
@@ -3435,31 +3764,47 @@ long wxListMainWindow::GetNextItem( long item,
 // deleting stuff
 // ----------------------------------------------------------------------------
 
-void wxListMainWindow::DeleteItem( long index )
+void wxListMainWindow::DeleteItem( long lindex )
 {
     size_t count = GetItemCount();
 
-    wxCHECK_RET( (index >= 0) && ((size_t)index < count),
+    wxCHECK_RET( (lindex >= 0) && ((size_t)lindex < count),
                  _T("invalid item index in DeleteItem") );
 
+    size_t index = (size_t)lindex;
+
     m_dirty = TRUE;
 
     // select the next item when the selected one is deleted
-    if ( m_current == (size_t)index )
+    if ( m_current == index )
     {
         // the last valid index after deleting the item will be count-2
-        if ( ++m_current >= count - 2 )
+        if ( m_current == count - 1 )
         {
-            m_current = count - 2;
+            m_current--;
         }
     }
 
-    SendNotify( (size_t)index, wxEVT_COMMAND_LIST_DELETE_ITEM );
+    SendNotify( index, wxEVT_COMMAND_LIST_DELETE_ITEM );
 
-    if ( !IsVirtual() )
+    if ( InReportView() )
     {
-        m_lines.RemoveAt( (size_t)index );
+        ResetVisibleLinesRange();
     }
+
+    if ( IsVirtual() )
+    {
+        m_countVirt--;
+
+        m_selStore.OnItemDelete(index);
+    }
+    else
+    {
+        m_lines.RemoveAt( index );
+    }
+
+    m_dirty = TRUE;
+    RefreshAfter(index);
 }
 
 void wxListMainWindow::DeleteColumn( int col )
@@ -3481,7 +3826,8 @@ void wxListMainWindow::DeleteAllItems()
     }
 
     m_dirty = TRUE;
-    m_current = (size_t)-1;
+
+    ResetCurrent();
 
     // to make the deletion of all items faster, we don't send the
     // notifications for each item deletion in this case but only one event
@@ -3492,9 +3838,21 @@ void wxListMainWindow::DeleteAllItems()
     event.SetEventObject( GetParent() );
     GetParent()->GetEventHandler()->ProcessEvent( event );
 
+    if ( IsVirtual() )
+    {
+        m_countVirt = 0;
+
+        ResetVisibleLinesRange();
+    }
+
+    if ( InReportView() )
+    {
+        ResetVisibleLinesRange();
+    }
+
     m_lines.Clear();
 
-    m_selections.Clear();
+    m_selStore.Clear();
 }
 
 void wxListMainWindow::DeleteEverything()
@@ -3565,15 +3923,23 @@ long wxListMainWindow::HitTest( int x, int y, int &flags )
 {
     CalcUnscrolledPosition( x, y, &x, &y );
 
-    size_t count = GetItemCount();
-    for (size_t i = 0; i < count; i++)
+    if ( HasFlag(wxLC_REPORT) )
     {
-        wxListLineData *line = GetLine(i);
-        long ret = line->IsHit( x, y );
-        if (ret)
+        size_t current = y / GetLineHeight();
+        flags = HitTestLine(current, x, y);
+        if ( flags )
+            return current;
+    }
+    else // !report
+    {
+        // TODO: optimize it too! this is less simple than for report view but
+        //       enumerating all items is still not a way to do it!!
+        size_t count = GetItemCount();
+        for ( size_t current = 0; current < count; current++ )
         {
-            flags = (int)ret;
-            return i;
+            flags = HitTestLine(current, x, y);
+            if ( flags )
+                return current;
         }
     }
 
@@ -3610,11 +3976,14 @@ void wxListMainWindow::InsertItem( wxListItem &item )
         wxFAIL_MSG( _T("unknown mode") );
     }
 
-    wxListLineData *line = new wxListLineData( this, id );
+    wxListLineData *line = new wxListLineData(this);
 
     line->SetItem( 0, item );
 
     m_lines.Insert( line, id );
+
+    m_dirty = TRUE;
+    RefreshLines(id, GetItemCount() - 1);
 }
 
 void wxListMainWindow::InsertColumn( long col, wxListItem &item )
@@ -3670,8 +4039,9 @@ void wxListMainWindow::SortItems( wxListCtrlCompare fn, long data )
 
 void wxListMainWindow::OnScroll(wxScrollWinEvent& event)
 {
-    // update our idea of which lines are shown before scrolling the window
-    UpdateShownLinesRange();
+    // update our idea of which lines are shown when we redraw the window the
+    // next time
+    ResetVisibleLinesRange();
 
     // FIXME
 #if defined(__WXGTK__) && !defined(__WXUNIVERSAL__)
@@ -3692,29 +4062,53 @@ void wxListMainWindow::OnScroll(wxScrollWinEvent& event)
     }
 }
 
-void wxListMainWindow::UpdateShownLinesRange()
+int wxListMainWindow::GetCountPerPage() const
 {
-    // only optimize redrawing for the report view using m_lineFrom/To
-    if ( !HasFlag(wxLC_REPORT) )
-        return;
-
-    size_t lineFromOld = m_lineFrom,
-           lineToOld = m_lineTo;
+    if ( !m_linesPerPage )
+    {
+        wxConstCast(this, wxListMainWindow)->
+            m_linesPerPage = GetClientSize().y / GetLineHeight();
+    }
 
-    m_lineFrom = GetScrollPos(wxVERTICAL);
+    return m_linesPerPage;
+}
 
-    size_t count = GetItemCount();
+void wxListMainWindow::GetVisibleLinesRange(size_t *from, size_t *to)
+{
+    wxASSERT_MSG( HasFlag(wxLC_REPORT), _T("this is for report mode only") );
 
-    wxASSERT_MSG( m_lineFrom < count, _T("invalid scroll position?") );
+    if ( m_lineFrom == (size_t)-1 )
+    {
+        size_t count = GetItemCount();
+        if ( count )
+        {
+            m_lineFrom = GetScrollPos(wxVERTICAL);
 
-    m_lineTo = m_lineFrom + m_linesPerPage - 1;
-    if ( m_lineTo >= count )
-        m_lineTo = count - 1;
+            // this may happen if SetScrollbars() hadn't been called yet
+            if ( m_lineFrom >= count )
+                m_lineFrom = count - 1;
 
-    if ( m_lineFrom != lineFromOld || m_lineTo != lineToOld )
-    {
-        m_dirty = TRUE;
+            // we redraw one extra line but this is needed to make the redrawing
+            // logic work when there is a fractional number of lines on screen
+            m_lineTo = m_lineFrom + m_linesPerPage;
+            if ( m_lineTo >= count )
+                m_lineTo = count - 1;
+        }
+        else // empty control
+        {
+            m_lineFrom = 0;
+            m_lineTo = (size_t)-1;
+        }
     }
+
+    wxASSERT_MSG( IsEmpty() ||
+                  (m_lineFrom <= m_lineTo && m_lineTo < GetItemCount()),
+                  _T("GetVisibleLinesRange() returns incorrect result") );
+
+    if ( from )
+        *from = m_lineFrom;
+    if ( to )
+        *to = m_lineTo;
 }
 
 // -------------------------------------------------------------------------------------
@@ -3837,6 +4231,9 @@ wxListCtrl::wxListCtrl()
 
 wxListCtrl::~wxListCtrl()
 {
+    if ( m_mainWin )
+        m_mainWin->ResetCurrent();
+
     if (m_ownsImageListNormal)
         delete m_imageListNormal;
     if (m_ownsImageListSmall)
@@ -3874,7 +4271,7 @@ bool wxListCtrl::Create(wxWindow *parent,
     m_mainWin = (wxListMainWindow*) NULL;
     m_headerWin = (wxListHeaderWindow*) NULL;
 
-    if ( !(style & (wxLC_REPORT | wxLC_LIST | wxLC_ICON)) )
+    if ( !(style & wxLC_MASK_TYPE) )
     {
         style = style | wxLC_LIST;
     }
@@ -3882,16 +4279,16 @@ bool wxListCtrl::Create(wxWindow *parent,
     if ( !wxControl::Create( parent, id, pos, size, style, validator, name ) )
         return FALSE;
 
-    if ( style & wxSUNKEN_BORDER )
-        style -= wxSUNKEN_BORDER;
+    // don't create the inner window with the border
+    style &= ~wxSUNKEN_BORDER;
 
     m_mainWin = new wxListMainWindow( this, -1, wxPoint(0,0), size, style );
 
-    if (HasFlag(wxLC_REPORT))
+    if ( HasFlag(wxLC_REPORT) )
     {
         CreateHeaderWindow();
 
-        if (HasFlag(wxLC_NO_HEADER))
+        if ( HasFlag(wxLC_NO_HEADER) )
         {
             // VZ: why do we create it at all then?
             m_headerWin->Show( FALSE );
@@ -3901,21 +4298,17 @@ bool wxListCtrl::Create(wxWindow *parent,
     return TRUE;
 }
 
-void wxListCtrl::OnSize( wxSizeEvent &WXUNUSED(event) )
-{
-    /* handled in OnIdle */
-
-    if (m_mainWin) m_mainWin->m_dirty = TRUE;
-}
-
 void wxListCtrl::SetSingleStyle( long style, bool add )
 {
+    wxASSERT_MSG( !(style & wxLC_VIRTUAL),
+                  _T("wxLC_VIRTUAL can't be [un]set") );
+
     long flag = GetWindowStyle();
 
     if (add)
     {
         if (style & wxLC_MASK_TYPE)
-            flag &= ~wxLC_MASK_TYPE;
+            flag &= ~(wxLC_MASK_TYPE | wxLC_VIRTUAL);
         if (style & wxLC_MASK_ALIGN)
             flag &= ~wxLC_MASK_ALIGN;
         if (style & wxLC_MASK_SORT)
@@ -4352,17 +4745,15 @@ bool wxListCtrl::SortItems( wxListCtrlCompare fn, long data )
 }
 
 // ----------------------------------------------------------------------------
-// OnIdle() handler: this is the place where we redraw the control
+// event handlers
 // ----------------------------------------------------------------------------
 
-void wxListCtrl::OnIdle( wxIdleEvent & WXUNUSED(event) )
+void wxListCtrl::OnSize(wxSizeEvent& event)
 {
-    // do it only if needed
-    if ( !m_mainWin->m_dirty )
+    if ( !m_mainWin )
         return;
 
-    int cw = 0;
-    int ch = 0;
+    int cw, ch;
     GetClientSize( &cw, &ch );
 
     if ( m_mainWin->HasHeader() )
@@ -4375,16 +4766,18 @@ void wxListCtrl::OnIdle( wxIdleEvent & WXUNUSED(event) )
         m_mainWin->SetSize( 0, 0, cw, ch );
     }
 
-    m_mainWin->CalculatePositions();
+    m_mainWin->RecalculatePositions();
+}
 
-    m_mainWin->m_dirty = FALSE;
-    m_mainWin->Refresh();
+void wxListCtrl::OnIdle( wxIdleEvent & event )
+{
+    event.Skip();
 
-    if ( m_headerWin && m_headerWin->m_dirty )
-    {
-        m_headerWin->m_dirty = FALSE;
-        m_headerWin->Refresh();
-    }
+    // do it only if needed
+    if ( !m_mainWin->m_dirty )
+        return;
+
+    m_mainWin->RecalculatePositions();
 }
 
 // ----------------------------------------------------------------------------
@@ -4437,9 +4830,6 @@ bool wxListCtrl::SetFont( const wxFont &font )
         m_headerWin->SetFont( font );
     }
 
-    // invalidate it as the font changed
-    m_mainWin->OnFontChange();
-
     return TRUE;
 }
 
@@ -4514,6 +4904,15 @@ int wxListCtrl::OnGetItemImage(long item) const
     return -1;
 }
 
+wxListItemAttr *wxListCtrl::OnGetItemAttr(long item) const
+{
+    wxASSERT_MSG( item >= 0 && item < GetItemCount(),
+                  _T("invalid item index in OnGetItemAttr()") );
+
+    // no attributes by default
+    return NULL;
+}
+
 void wxListCtrl::SetItemCount(long count)
 {
     wxASSERT_MSG( IsVirtual(), _T("this is for virtual controls only") );
@@ -4521,6 +4920,14 @@ void wxListCtrl::SetItemCount(long count)
     m_mainWin->SetItemCount(count);
 }
 
-#endif // wxUSE_LISTCTRL
+void wxListCtrl::RefreshItem(long item)
+{
+    m_mainWin->RefreshLine(item);
+}
 
-#endif
+void wxListCtrl::RefreshItems(long itemFrom, long itemTo)
+{
+    m_mainWin->RefreshLines(itemFrom, itemTo);
+}
+
+#endif // wxUSE_LISTCTRL