git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@24592 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
		
			
				
	
	
		
			4837 lines
		
	
	
		
			147 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			4837 lines
		
	
	
		
			147 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/////////////////////////////////////////////////////////////////////////////
 | 
						|
// Name:        treelistctrl.cpp (derived by treectlg.h)
 | 
						|
// Purpose:     multi column tree control implementation
 | 
						|
// Author:      Robert Roebling
 | 
						|
// Created:     01/02/97
 | 
						|
// Modified:    Alberto Griggio, 2002
 | 
						|
//              22/10/98 - almost total rewrite, simpler interface (VZ)
 | 
						|
// Id:          $Id$
 | 
						|
// Copyright:   (c) 1998 Robert Roebling, Julian Smart and Markus Holzem
 | 
						|
// Licence:     wxWindows licence
 | 
						|
/////////////////////////////////////////////////////////////////////////////
 | 
						|
 | 
						|
// ===========================================================================
 | 
						|
// declarations
 | 
						|
// ===========================================================================
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// headers
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
 | 
						|
#if defined(__GNUG__) && !defined(__APPLE__)
 | 
						|
  #pragma implementation "treelistctrl.h"
 | 
						|
#endif
 | 
						|
 | 
						|
// For compilers that support precompilation, includes "wx.h".
 | 
						|
#include "wx/wxprec.h"
 | 
						|
 | 
						|
#ifdef __BORLANDC__
 | 
						|
    #pragma hdrstop
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
#include <wx/treebase.h>
 | 
						|
#include <wx/timer.h>
 | 
						|
#include <wx/textctrl.h>
 | 
						|
#include <wx/imaglist.h>
 | 
						|
#include <wx/settings.h>
 | 
						|
#include <wx/dcclient.h>
 | 
						|
#include <wx/dcscreen.h>
 | 
						|
#include <wx/scrolwin.h>
 | 
						|
 | 
						|
//#include "wx/gizmos/treelistctrl.h"
 | 
						|
#include "treelistctrl.h"
 | 
						|
 | 
						|
 | 
						|
#ifdef __WXGTK__
 | 
						|
    #include <gtk/gtk.h>
 | 
						|
    #include <wx/gtk/win_gtk.h>
 | 
						|
#endif
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// array types
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
 | 
						|
class  wxTreeListItem;
 | 
						|
 | 
						|
WX_DEFINE_ARRAY(wxTreeListItem *, wxArrayTreeListItems);
 | 
						|
 | 
						|
#include <wx/dynarray.h>
 | 
						|
WX_DECLARE_OBJARRAY(wxTreeListColumnInfo, wxArrayTreeListColumnInfo);
 | 
						|
#include <wx/arrimpl.cpp>
 | 
						|
WX_DEFINE_OBJARRAY(wxArrayTreeListColumnInfo);
 | 
						|
 | 
						|
#if !wxCHECK_VERSION(2, 3, 3)
 | 
						|
WX_DEFINE_ARRAY(short, wxArrayShort);
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
// --------------------------------------------------------------------------
 | 
						|
// constants
 | 
						|
// --------------------------------------------------------------------------
 | 
						|
 | 
						|
static const int NO_IMAGE = -1;
 | 
						|
 | 
						|
#define PIXELS_PER_UNIT 10
 | 
						|
 | 
						|
const wxChar* wxTreeListCtrlNameStr = wxT("treelistctrl");
 | 
						|
 | 
						|
static wxTreeListColumnInfo wxInvalidTreeListColumnInfo;
 | 
						|
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// private classes
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
//  wxTreeListHeaderWindow (internal)
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
 | 
						|
class  wxTreeListHeaderWindow : public wxWindow
 | 
						|
{
 | 
						|
protected:
 | 
						|
    wxTreeListMainWindow    *m_owner;
 | 
						|
    wxCursor          *m_currentCursor;
 | 
						|
    wxCursor          *m_resizeCursor;
 | 
						|
    bool               m_isDragging;
 | 
						|
 | 
						|
    // column being resized
 | 
						|
    int m_column;
 | 
						|
 | 
						|
    // divider line position in logical (unscrolled) coords
 | 
						|
    int m_currentX;
 | 
						|
 | 
						|
    // minimal position beyond which the divider line can't be dragged in
 | 
						|
    // logical coords
 | 
						|
    int m_minX;
 | 
						|
 | 
						|
    wxArrayTreeListColumnInfo m_columns;
 | 
						|
 | 
						|
    // total width of the columns
 | 
						|
    int m_total_col_width;
 | 
						|
 | 
						|
 | 
						|
public:
 | 
						|
    wxTreeListHeaderWindow();
 | 
						|
 | 
						|
    wxTreeListHeaderWindow( wxWindow *win,
 | 
						|
                        wxWindowID id,
 | 
						|
                        wxTreeListMainWindow *owner,
 | 
						|
                        const wxPoint &pos = wxDefaultPosition,
 | 
						|
                        const wxSize &size = wxDefaultSize,
 | 
						|
                        long style = 0,
 | 
						|
                        const wxString &name = wxT("wxtreelistctrlcolumntitles") );
 | 
						|
 | 
						|
    virtual ~wxTreeListHeaderWindow();
 | 
						|
 | 
						|
    void DoDrawRect( wxDC *dc, int x, int y, int w, int h );
 | 
						|
    void DrawCurrent();
 | 
						|
    void AdjustDC(wxDC& dc);
 | 
						|
 | 
						|
    void OnPaint( wxPaintEvent &event );
 | 
						|
    void OnMouse( wxMouseEvent &event );
 | 
						|
    void OnSetFocus( wxFocusEvent &event );
 | 
						|
 | 
						|
 | 
						|
    // columns manipulation
 | 
						|
 | 
						|
    size_t GetColumnCount() const { return m_columns.GetCount(); }
 | 
						|
 | 
						|
    void AddColumn(const wxTreeListColumnInfo& col);
 | 
						|
 | 
						|
    void InsertColumn(size_t before, const wxTreeListColumnInfo& col);
 | 
						|
 | 
						|
    void RemoveColumn(size_t column);
 | 
						|
 | 
						|
    void SetColumn(size_t column, const wxTreeListColumnInfo& info);
 | 
						|
    const wxTreeListColumnInfo& GetColumn(size_t column) const
 | 
						|
    {
 | 
						|
        wxCHECK_MSG(column < GetColumnCount(), wxInvalidTreeListColumnInfo, wxT("Invalid column"));
 | 
						|
        return m_columns[column];
 | 
						|
    }
 | 
						|
    wxTreeListColumnInfo& GetColumn(size_t column)
 | 
						|
    {
 | 
						|
        wxCHECK_MSG(column < GetColumnCount(), wxInvalidTreeListColumnInfo, wxT("Invalid column"));
 | 
						|
        return m_columns[column];
 | 
						|
    }
 | 
						|
 | 
						|
    void SetColumnWidth(size_t column, size_t width);
 | 
						|
 | 
						|
    void SetColumnText(size_t column, const wxString& text)
 | 
						|
    {
 | 
						|
        wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
 | 
						|
        m_columns[column].SetText(text);
 | 
						|
    }
 | 
						|
 | 
						|
    wxString GetColumnText(size_t column) const
 | 
						|
    {
 | 
						|
        wxCHECK_MSG(column < GetColumnCount(), wxEmptyString, wxT("Invalid column"));
 | 
						|
        return m_columns[column].GetText();
 | 
						|
    }
 | 
						|
 | 
						|
    int GetColumnWidth(size_t column) const
 | 
						|
    {
 | 
						|
        wxCHECK_MSG(column < GetColumnCount(), -1, wxT("Invalid column"));
 | 
						|
        return m_columns[column].GetWidth();
 | 
						|
    }
 | 
						|
 | 
						|
    int GetWidth() const { return m_total_col_width; }
 | 
						|
 | 
						|
    // needs refresh
 | 
						|
    bool m_dirty;
 | 
						|
 | 
						|
private:
 | 
						|
    // common part of all ctors
 | 
						|
    void Init();
 | 
						|
 | 
						|
    void SendListEvent(wxEventType type, wxPoint pos);
 | 
						|
 | 
						|
    DECLARE_DYNAMIC_CLASS(wxTreeListHeaderWindow)
 | 
						|
    DECLARE_EVENT_TABLE()
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
// this is the "true" control
 | 
						|
class  wxTreeListMainWindow: public wxScrolledWindow
 | 
						|
{
 | 
						|
public:
 | 
						|
    // creation
 | 
						|
    // --------
 | 
						|
    wxTreeListMainWindow() { Init(); }
 | 
						|
 | 
						|
    wxTreeListMainWindow(wxTreeListCtrl *parent, wxWindowID id = -1,
 | 
						|
               const wxPoint& pos = wxDefaultPosition,
 | 
						|
               const wxSize& size = wxDefaultSize,
 | 
						|
               long style = wxTR_DEFAULT_STYLE,
 | 
						|
               const wxValidator &validator = wxDefaultValidator,
 | 
						|
               const wxString& name = wxT("wxtreelistmainwindow"))
 | 
						|
    {
 | 
						|
        Init();
 | 
						|
        Create(parent, id, pos, size, style, validator, name);
 | 
						|
    }
 | 
						|
 | 
						|
    virtual ~wxTreeListMainWindow();
 | 
						|
 | 
						|
    bool Create(wxTreeListCtrl *parent, wxWindowID id = -1,
 | 
						|
                const wxPoint& pos = wxDefaultPosition,
 | 
						|
                const wxSize& size = wxDefaultSize,
 | 
						|
                long style = wxTR_DEFAULT_STYLE,
 | 
						|
                const wxValidator &validator = wxDefaultValidator,
 | 
						|
                const wxString& name = wxT("wxtreelistctrl"));
 | 
						|
 | 
						|
    // accessors
 | 
						|
    // ---------
 | 
						|
 | 
						|
        // get the total number of items in the control
 | 
						|
    size_t GetCount() const;
 | 
						|
 | 
						|
        // indent is the number of pixels the children are indented relative to
 | 
						|
        // the parents position. SetIndent() also redraws the control
 | 
						|
        // immediately.
 | 
						|
    unsigned int GetIndent() const { return m_indent; }
 | 
						|
    void SetIndent(unsigned int indent);
 | 
						|
 | 
						|
        // spacing is the number of pixels between the start and the Text
 | 
						|
    unsigned int GetSpacing() const { return m_spacing; }
 | 
						|
    void SetSpacing(unsigned int spacing);
 | 
						|
 | 
						|
    // see wxTreeListCtrl for the meaning
 | 
						|
    unsigned int GetLineSpacing() const { return m_linespacing; }
 | 
						|
    void SetLineSpacing(unsigned int spacing);
 | 
						|
 | 
						|
        // image list: these functions allow to associate an image list with
 | 
						|
        // the control and retrieve it. Note that when assigned with
 | 
						|
        // SetImageList, the control does _not_ delete
 | 
						|
        // the associated image list when it's deleted in order to allow image
 | 
						|
        // lists to be shared between different controls. If you use
 | 
						|
        // AssignImageList, the control _does_ delete the image list.
 | 
						|
        //
 | 
						|
        // The normal image list is for the icons which correspond to the
 | 
						|
        // normal tree item state (whether it is selected or not).
 | 
						|
        // Additionally, the application might choose to show a state icon
 | 
						|
        // which corresponds to an app-defined item state (for example,
 | 
						|
        // checked/unchecked) which are taken from the state image list.
 | 
						|
    wxImageList *GetImageList() const;
 | 
						|
    wxImageList *GetStateImageList() const;
 | 
						|
    wxImageList *GetButtonsImageList() const;
 | 
						|
 | 
						|
    void SetImageList(wxImageList *imageList);
 | 
						|
    void SetStateImageList(wxImageList *imageList);
 | 
						|
    void SetButtonsImageList(wxImageList *imageList);
 | 
						|
    void AssignImageList(wxImageList *imageList);
 | 
						|
    void AssignStateImageList(wxImageList *imageList);
 | 
						|
    void AssignButtonsImageList(wxImageList *imageList);
 | 
						|
 | 
						|
    // Functions to work with tree ctrl items.
 | 
						|
 | 
						|
    // accessors
 | 
						|
    // ---------
 | 
						|
 | 
						|
        // retrieve item's label
 | 
						|
    wxString GetItemText(const wxTreeItemId& item) const
 | 
						|
    { return GetItemText(item, GetMainColumn()); }
 | 
						|
        // get one of the images associated with the item (normal by default)
 | 
						|
    int GetItemImage(const wxTreeItemId& item,
 | 
						|
                     wxTreeItemIcon which = wxTreeItemIcon_Normal) const
 | 
						|
    { return GetItemImage(item, GetMainColumn(), which); }
 | 
						|
 | 
						|
        // get the data associated with the item
 | 
						|
    wxTreeItemData *GetItemData(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
    bool GetItemBold(const wxTreeItemId& item) const;
 | 
						|
    wxColour GetItemTextColour(const wxTreeItemId& item) const;
 | 
						|
    wxColour GetItemBackgroundColour(const wxTreeItemId& item) const;
 | 
						|
    wxFont GetItemFont(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
    // modifiers
 | 
						|
    // ---------
 | 
						|
 | 
						|
        // set item's label
 | 
						|
    void SetItemText(const wxTreeItemId& item, const wxString& text)
 | 
						|
    { SetItemText(item, GetMainColumn(), text); }
 | 
						|
 | 
						|
        // get one of the images associated with the item (normal by default)
 | 
						|
    void SetItemImage(const wxTreeItemId& item, int image,
 | 
						|
                      wxTreeItemIcon which = wxTreeItemIcon_Normal)
 | 
						|
    { SetItemImage(item, GetMainColumn(), image, which); }
 | 
						|
 | 
						|
        // associate some data with the item
 | 
						|
    void SetItemData(const wxTreeItemId& item, wxTreeItemData *data);
 | 
						|
 | 
						|
        // force appearance of [+] button near the item. This is useful to
 | 
						|
        // allow the user to expand the items which don't have any children now
 | 
						|
        // - but instead add them only when needed, thus minimizing memory
 | 
						|
        // usage and loading time.
 | 
						|
    void SetItemHasChildren(const wxTreeItemId& item, bool has = TRUE);
 | 
						|
 | 
						|
        // the item will be shown in bold
 | 
						|
    void SetItemBold(const wxTreeItemId& item, bool bold = TRUE);
 | 
						|
 | 
						|
        // set the item's text colour
 | 
						|
    void SetItemTextColour(const wxTreeItemId& item, const wxColour& col);
 | 
						|
 | 
						|
        // set the item's background colour
 | 
						|
    void SetItemBackgroundColour(const wxTreeItemId& item,
 | 
						|
                                 const wxColour& col);
 | 
						|
 | 
						|
        // set the item's font (should be of the same height for all items)
 | 
						|
    void SetItemFont(const wxTreeItemId& item, const wxFont& font);
 | 
						|
 | 
						|
        // set the window font
 | 
						|
    virtual bool SetFont( const wxFont &font );
 | 
						|
 | 
						|
       // set the styles.  No need to specify a GetWindowStyle here since
 | 
						|
       // the base wxWindow member function will do it for us
 | 
						|
    void SetWindowStyle(const long styles);
 | 
						|
 | 
						|
    // item status inquiries
 | 
						|
    // ---------------------
 | 
						|
 | 
						|
        // is the item visible (it might be outside the view or not expanded)?
 | 
						|
    bool IsVisible(const wxTreeItemId& item) const;
 | 
						|
        // does the item has any children?
 | 
						|
    bool HasChildren(const wxTreeItemId& item) const
 | 
						|
      { return ItemHasChildren(item); }
 | 
						|
    bool ItemHasChildren(const wxTreeItemId& item) const;
 | 
						|
        // is the item expanded (only makes sense if HasChildren())?
 | 
						|
    bool IsExpanded(const wxTreeItemId& item) const;
 | 
						|
        // is this item currently selected (the same as has focus)?
 | 
						|
    bool IsSelected(const wxTreeItemId& item) const;
 | 
						|
        // is item text in bold font?
 | 
						|
    bool IsBold(const wxTreeItemId& item) const;
 | 
						|
        // does the layout include space for a button?
 | 
						|
 | 
						|
    // number of children
 | 
						|
    // ------------------
 | 
						|
 | 
						|
        // if 'recursively' is FALSE, only immediate children count, otherwise
 | 
						|
        // the returned number is the number of all items in this branch
 | 
						|
    size_t GetChildrenCount(const wxTreeItemId& item, bool recursively = TRUE);
 | 
						|
 | 
						|
    // navigation
 | 
						|
    // ----------
 | 
						|
 | 
						|
    // wxTreeItemId.IsOk() will return FALSE if there is no such item
 | 
						|
 | 
						|
        // get the root tree item
 | 
						|
    wxTreeItemId GetRootItem() const { return m_anchor; }
 | 
						|
 | 
						|
        // get the item currently selected (may return NULL if no selection)
 | 
						|
    wxTreeItemId GetSelection() const { return m_current; }
 | 
						|
 | 
						|
        // get the items currently selected, return the number of such item
 | 
						|
    size_t GetSelections(wxArrayTreeItemIds&) const;
 | 
						|
 | 
						|
        // get the parent of this item (may return NULL if root)
 | 
						|
    wxTreeItemId GetParent(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
        // for this enumeration function you must pass in a "cookie" parameter
 | 
						|
        // which is opaque for the application but is necessary for the library
 | 
						|
        // to make these functions reentrant (i.e. allow more than one
 | 
						|
        // enumeration on one and the same object simultaneously). Of course,
 | 
						|
        // the "cookie" passed to GetFirstChild() and GetNextChild() should be
 | 
						|
        // the same!
 | 
						|
 | 
						|
        // get the first child of this item
 | 
						|
    wxTreeItemId GetFirstChild(const wxTreeItemId& item, long& cookie) const;
 | 
						|
        // get the next child
 | 
						|
    wxTreeItemId GetNextChild(const wxTreeItemId& item, long& cookie) const;
 | 
						|
        // get the last child of this item - this method doesn't use cookies
 | 
						|
    wxTreeItemId GetLastChild(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
        // get the next sibling of this item
 | 
						|
    wxTreeItemId GetNextSibling(const wxTreeItemId& item) const;
 | 
						|
        // get the previous sibling
 | 
						|
    wxTreeItemId GetPrevSibling(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
        // get first visible item
 | 
						|
    wxTreeItemId GetFirstVisibleItem() const;
 | 
						|
        // get the next visible item: item must be visible itself!
 | 
						|
        // see IsVisible() and wxTreeCtrl::GetFirstVisibleItem()
 | 
						|
    wxTreeItemId GetNextVisible(const wxTreeItemId& item) const;
 | 
						|
        // get the previous visible item: item must be visible itself!
 | 
						|
    wxTreeItemId GetPrevVisible(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
        // Only for internal use right now, but should probably be public
 | 
						|
    wxTreeItemId GetNext(const wxTreeItemId& item) const;
 | 
						|
 | 
						|
    // operations
 | 
						|
    // ----------
 | 
						|
 | 
						|
        // add the root node to the tree
 | 
						|
    wxTreeItemId AddRoot(const wxString& text,
 | 
						|
                         int image = -1, int selectedImage = -1,
 | 
						|
                         wxTreeItemData *data = NULL);
 | 
						|
 | 
						|
        // insert a new item in as the first child of the parent
 | 
						|
    wxTreeItemId PrependItem(const wxTreeItemId& parent,
 | 
						|
                             const wxString& text,
 | 
						|
                             int image = -1, int selectedImage = -1,
 | 
						|
                             wxTreeItemData *data = NULL);
 | 
						|
 | 
						|
        // insert a new item after a given one
 | 
						|
    wxTreeItemId InsertItem(const wxTreeItemId& parent,
 | 
						|
                            const wxTreeItemId& idPrevious,
 | 
						|
                            const wxString& text,
 | 
						|
                            int image = -1, int selectedImage = -1,
 | 
						|
                            wxTreeItemData *data = NULL);
 | 
						|
 | 
						|
        // insert a new item before the one with the given index
 | 
						|
    wxTreeItemId InsertItem(const wxTreeItemId& parent,
 | 
						|
                            size_t index,
 | 
						|
                            const wxString& text,
 | 
						|
                            int image = -1, int selectedImage = -1,
 | 
						|
                            wxTreeItemData *data = NULL);
 | 
						|
 | 
						|
        // insert a new item in as the last child of the parent
 | 
						|
    wxTreeItemId AppendItem(const wxTreeItemId& parent,
 | 
						|
                            const wxString& text,
 | 
						|
                            int image = -1, int selectedImage = -1,
 | 
						|
                            wxTreeItemData *data = NULL);
 | 
						|
 | 
						|
        // delete this item and associated data if any
 | 
						|
    void Delete(const wxTreeItemId& item);
 | 
						|
        // delete all children (but don't delete the item itself)
 | 
						|
        // NB: this won't send wxEVT_COMMAND_TREE_ITEM_DELETED events
 | 
						|
    void DeleteChildren(const wxTreeItemId& item);
 | 
						|
        // delete all items from the tree
 | 
						|
        // NB: this won't send wxEVT_COMMAND_TREE_ITEM_DELETED events
 | 
						|
    void DeleteAllItems();
 | 
						|
 | 
						|
        // expand this item
 | 
						|
    void Expand(const wxTreeItemId& item);
 | 
						|
        // expand this item and all subitems recursively
 | 
						|
    void ExpandAll(const wxTreeItemId& item);
 | 
						|
        // collapse the item without removing its children
 | 
						|
    void Collapse(const wxTreeItemId& item);
 | 
						|
        // collapse the item and remove all children
 | 
						|
    void CollapseAndReset(const wxTreeItemId& item);
 | 
						|
        // toggles the current state
 | 
						|
    void Toggle(const wxTreeItemId& item);
 | 
						|
 | 
						|
        // remove the selection from currently selected item (if any)
 | 
						|
    void Unselect();
 | 
						|
    void UnselectAll();
 | 
						|
        // select this item
 | 
						|
    void SelectItem(const wxTreeItemId& item, bool unselect_others=TRUE,
 | 
						|
                    bool extended_select=FALSE);
 | 
						|
        // make sure this item is visible (expanding the parent item and/or
 | 
						|
        // scrolling to this item if necessary)
 | 
						|
    void EnsureVisible(const wxTreeItemId& item);
 | 
						|
        // scroll to this item (but don't expand its parent)
 | 
						|
    void ScrollTo(const wxTreeItemId& item);
 | 
						|
    void AdjustMyScrollbars();
 | 
						|
 | 
						|
        // The first function is more portable (because easier to implement
 | 
						|
        // on other platforms), but the second one returns some extra info.
 | 
						|
    wxTreeItemId HitTest(const wxPoint& point)
 | 
						|
        { int dummy; return HitTest(point, dummy); }
 | 
						|
    wxTreeItemId HitTest(const wxPoint& point, int& flags)
 | 
						|
    { int col; return HitTest(point, flags, col); }
 | 
						|
    // ALB
 | 
						|
    wxTreeItemId HitTest(const wxPoint& point, int& flags, int& column);
 | 
						|
 | 
						|
 | 
						|
        // get the bounding rectangle of the item (or of its label only)
 | 
						|
    bool GetBoundingRect(const wxTreeItemId& item,
 | 
						|
                         wxRect& rect,
 | 
						|
                         bool textOnly = FALSE) const;
 | 
						|
 | 
						|
        // Start editing the item label: this (temporarily) replaces the item
 | 
						|
        // with a one line edit control. The item will be selected if it hadn't
 | 
						|
        // been before.
 | 
						|
    void EditLabel( const wxTreeItemId& item ) { Edit( item ); }
 | 
						|
    void Edit( const wxTreeItemId& item );
 | 
						|
 | 
						|
    // sorting
 | 
						|
        // this function is called to compare 2 items and should return -1, 0
 | 
						|
        // or +1 if the first item is less than, equal to or greater than the
 | 
						|
        // second one. The base class version performs alphabetic comparaison
 | 
						|
        // of item labels (GetText)
 | 
						|
    virtual int OnCompareItems(const wxTreeItemId& item1,
 | 
						|
                               const wxTreeItemId& item2);
 | 
						|
        // sort the children of this item using OnCompareItems
 | 
						|
        //
 | 
						|
        // NB: this function is not reentrant and not MT-safe (FIXME)!
 | 
						|
    void SortChildren(const wxTreeItemId& item);
 | 
						|
 | 
						|
    // deprecated functions: use Set/GetItemImage directly
 | 
						|
        // get the selected item image
 | 
						|
    int GetItemSelectedImage(const wxTreeItemId& item) const
 | 
						|
        { return GetItemImage(item, wxTreeItemIcon_Selected); }
 | 
						|
        // set the selected item image
 | 
						|
    void SetItemSelectedImage(const wxTreeItemId& item, int image)
 | 
						|
        { SetItemImage(item, image, wxTreeItemIcon_Selected); }
 | 
						|
 | 
						|
    // implementation only from now on
 | 
						|
 | 
						|
    // overridden base class virtuals
 | 
						|
    virtual bool SetBackgroundColour(const wxColour& colour);
 | 
						|
    virtual bool SetForegroundColour(const wxColour& colour);
 | 
						|
 | 
						|
    // callbacks
 | 
						|
    void OnPaint( wxPaintEvent &event );
 | 
						|
    void OnSetFocus( wxFocusEvent &event );
 | 
						|
    void OnKillFocus( wxFocusEvent &event );
 | 
						|
    void OnChar( wxKeyEvent &event );
 | 
						|
    void OnMouse( wxMouseEvent &event );
 | 
						|
    void OnIdle( wxIdleEvent &event );
 | 
						|
    void OnSize(wxSizeEvent& event); // ALB
 | 
						|
    void OnScroll(wxScrollWinEvent& event); // ALB
 | 
						|
 | 
						|
    // implementation helpers
 | 
						|
    void SendDeleteEvent(wxTreeListItem *itemBeingDeleted);
 | 
						|
 | 
						|
    void DrawBorder(const wxTreeItemId& item);
 | 
						|
    void DrawLine(const wxTreeItemId& item, bool below);
 | 
						|
 | 
						|
    size_t GetColumnCount() const
 | 
						|
    { return m_owner->GetHeaderWindow()->GetColumnCount(); }
 | 
						|
 | 
						|
    void SetMainColumn(size_t column)
 | 
						|
    {
 | 
						|
        if(column < GetColumnCount())
 | 
						|
            m_main_column = column;
 | 
						|
    }
 | 
						|
    size_t GetMainColumn() const { return m_main_column; }
 | 
						|
 | 
						|
    void SetItemText(const wxTreeItemId& item, size_t column,
 | 
						|
                     const wxString& text);
 | 
						|
    wxString GetItemText(const wxTreeItemId& item, size_t column) const;
 | 
						|
 | 
						|
    void SetItemImage(const wxTreeItemId& item, size_t column, int image,
 | 
						|
                      wxTreeItemIcon which = wxTreeItemIcon_Normal);
 | 
						|
    int GetItemImage(const wxTreeItemId& item, size_t column,
 | 
						|
                     wxTreeItemIcon which = wxTreeItemIcon_Normal) const;
 | 
						|
protected:
 | 
						|
    wxTreeListCtrl* m_owner; // ALB
 | 
						|
 | 
						|
    size_t m_main_column; // ALB
 | 
						|
 | 
						|
    friend class wxTreeListItem;
 | 
						|
    friend class wxTreeListRenameTimer;
 | 
						|
    friend class wxTreeListTextCtrl;
 | 
						|
 | 
						|
    wxFont               m_normalFont;
 | 
						|
    wxFont               m_boldFont;
 | 
						|
 | 
						|
    wxTreeListItem   *m_anchor;
 | 
						|
    wxTreeListItem   *m_current, *m_key_current, *m_currentEdit;
 | 
						|
    unsigned short       m_indent;
 | 
						|
    unsigned short       m_spacing;
 | 
						|
    int                  m_lineHeight;
 | 
						|
    unsigned short       m_linespacing;
 | 
						|
    wxPen                m_dottedPen;
 | 
						|
    wxBrush             *m_hilightBrush,
 | 
						|
                        *m_hilightUnfocusedBrush;
 | 
						|
    bool                 m_hasFocus;
 | 
						|
public:
 | 
						|
    bool                 m_dirty;
 | 
						|
protected:
 | 
						|
    bool                 m_ownsImageListNormal,
 | 
						|
                         m_ownsImageListState,
 | 
						|
                         m_ownsImageListButtons;
 | 
						|
    bool                 m_isDragging; // true between BEGIN/END drag events
 | 
						|
    bool                 m_renameAccept;
 | 
						|
    bool                 m_lastOnSame;  // last click on the same item as prev
 | 
						|
    wxImageList         *m_imageListNormal,
 | 
						|
                        *m_imageListState,
 | 
						|
                        *m_imageListButtons;
 | 
						|
 | 
						|
    int                  m_dragCount;
 | 
						|
    wxPoint              m_dragStart;
 | 
						|
    wxTreeListItem   *m_dropTarget;
 | 
						|
    wxCursor             m_oldCursor;  // cursor is changed while dragging
 | 
						|
    wxTreeListItem   *m_oldSelection;
 | 
						|
 | 
						|
    wxTimer             *m_renameTimer;
 | 
						|
    wxString             m_renameRes;
 | 
						|
 | 
						|
    // the common part of all ctors
 | 
						|
    void Init();
 | 
						|
 | 
						|
    // misc helpers
 | 
						|
    wxTreeItemId DoInsertItem(const wxTreeItemId& parent,
 | 
						|
                              size_t previous,
 | 
						|
                              const wxString& text,
 | 
						|
                              int image, int selectedImage,
 | 
						|
                              wxTreeItemData *data);
 | 
						|
    bool HasButtons(void) const
 | 
						|
        { return (m_imageListButtons != NULL)
 | 
						|
              || HasFlag(wxTR_TWIST_BUTTONS|wxTR_HAS_BUTTONS); }
 | 
						|
 | 
						|
protected:
 | 
						|
    void CalculateLineHeight();
 | 
						|
    int  GetLineHeight(wxTreeListItem *item) const;
 | 
						|
    void PaintLevel( wxTreeListItem *item, wxDC& dc, int level, int &y,
 | 
						|
                     int x_offset);
 | 
						|
    void PaintItem( wxTreeListItem *item, wxDC& dc);
 | 
						|
 | 
						|
    void CalculateLevel( wxTreeListItem *item, wxDC &dc, int level, int &y,
 | 
						|
                         int x_offset);
 | 
						|
    void CalculatePositions();
 | 
						|
    void CalculateSize( wxTreeListItem *item, wxDC &dc );
 | 
						|
 | 
						|
    void RefreshSubtree( wxTreeListItem *item );
 | 
						|
    void RefreshLine( wxTreeListItem *item );
 | 
						|
 | 
						|
    // redraw all selected items
 | 
						|
    void RefreshSelected();
 | 
						|
 | 
						|
    // RefreshSelected() recursive helper
 | 
						|
    void RefreshSelectedUnder(wxTreeListItem *item);
 | 
						|
 | 
						|
    void OnRenameTimer();
 | 
						|
    void OnRenameAccept();
 | 
						|
 | 
						|
    void FillArray(wxTreeListItem*, wxArrayTreeItemIds&) const;
 | 
						|
    void SelectItemRange( wxTreeListItem *item1, wxTreeListItem *item2 );
 | 
						|
    bool TagAllChildrenUntilLast(wxTreeListItem *crt_item,
 | 
						|
                                 wxTreeListItem *last_item, bool select);
 | 
						|
    bool TagNextChildren(wxTreeListItem *crt_item, wxTreeListItem *last_item,
 | 
						|
                         bool select);
 | 
						|
    void UnselectAllChildren( wxTreeListItem *item );
 | 
						|
 | 
						|
    void DrawDropEffect(wxTreeListItem *item);
 | 
						|
 | 
						|
private:
 | 
						|
    DECLARE_EVENT_TABLE()
 | 
						|
    DECLARE_DYNAMIC_CLASS(wxTreeListMainWindow)
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
// timer used for enabling in-place edit
 | 
						|
class  wxTreeListRenameTimer: public wxTimer
 | 
						|
{
 | 
						|
public:
 | 
						|
    wxTreeListRenameTimer( wxTreeListMainWindow *owner );
 | 
						|
 | 
						|
    void Notify();
 | 
						|
 | 
						|
private:
 | 
						|
    wxTreeListMainWindow   *m_owner;
 | 
						|
};
 | 
						|
 | 
						|
// control used for in-place edit
 | 
						|
class  wxTreeListTextCtrl: public wxTextCtrl
 | 
						|
{
 | 
						|
public:
 | 
						|
    wxTreeListTextCtrl( wxWindow *parent,
 | 
						|
                        const wxWindowID id,
 | 
						|
                        bool *accept,
 | 
						|
                        wxString *res,
 | 
						|
                        wxTreeListMainWindow *owner,
 | 
						|
                        const wxString &value = wxEmptyString,
 | 
						|
                        const wxPoint &pos = wxDefaultPosition,
 | 
						|
                        const wxSize &size = wxDefaultSize,
 | 
						|
                        int style = wxSIMPLE_BORDER,
 | 
						|
                        const wxValidator& validator = wxDefaultValidator,
 | 
						|
                        const wxString &name = wxTextCtrlNameStr );
 | 
						|
 | 
						|
    void OnChar( wxKeyEvent &event );
 | 
						|
    void OnKeyUp( wxKeyEvent &event );
 | 
						|
    void OnKillFocus( wxFocusEvent &event );
 | 
						|
 | 
						|
private:
 | 
						|
    bool               *m_accept;
 | 
						|
    wxString           *m_res;
 | 
						|
    wxTreeListMainWindow  *m_owner;
 | 
						|
    wxString            m_startValue;
 | 
						|
    bool                m_finished;
 | 
						|
 | 
						|
    DECLARE_EVENT_TABLE()
 | 
						|
};
 | 
						|
 | 
						|
// a tree item
 | 
						|
class  wxTreeListItem
 | 
						|
{
 | 
						|
public:
 | 
						|
    // ctors & dtor
 | 
						|
    wxTreeListItem() { m_data = NULL; }
 | 
						|
    wxTreeListItem( wxTreeListMainWindow *owner,
 | 
						|
                    wxTreeListItem *parent,
 | 
						|
                    const wxArrayString& text,
 | 
						|
                    int image,
 | 
						|
                    int selImage,
 | 
						|
                    wxTreeItemData *data );
 | 
						|
 | 
						|
    ~wxTreeListItem();
 | 
						|
 | 
						|
    // trivial accessors
 | 
						|
    wxArrayTreeListItems& GetChildren() { return m_children; }
 | 
						|
 | 
						|
    const wxString GetText() const
 | 
						|
    {
 | 
						|
        if(m_text.GetCount() > 0) return m_text[0];
 | 
						|
        return wxEmptyString;
 | 
						|
    }
 | 
						|
    const wxString GetText(size_t col) const
 | 
						|
    {
 | 
						|
        if(m_text.GetCount() > col) return m_text[col];
 | 
						|
        return wxEmptyString;
 | 
						|
    }
 | 
						|
    int GetImage(wxTreeItemIcon which = wxTreeItemIcon_Normal) const
 | 
						|
        { return m_images[which]; }
 | 
						|
    int GetImage(size_t col, wxTreeItemIcon which=wxTreeItemIcon_Normal) const
 | 
						|
    {
 | 
						|
        if(col == m_owner->GetMainColumn()) return m_images[which];
 | 
						|
        if(col < m_col_images.GetCount()) return m_col_images[col];
 | 
						|
        return NO_IMAGE;
 | 
						|
    }
 | 
						|
    wxTreeItemData *GetData() const { return m_data; }
 | 
						|
 | 
						|
    // returns the current image for the item (depending on its
 | 
						|
    // selected/expanded/whatever state)
 | 
						|
    int GetCurrentImage() const;
 | 
						|
 | 
						|
    void SetText( const wxString &text );
 | 
						|
    void SetText(size_t col, const wxString& text) // ALB
 | 
						|
    {
 | 
						|
        if(col < m_text.GetCount())
 | 
						|
            m_text[col] = text;
 | 
						|
        else if(col < m_owner->GetColumnCount()) {
 | 
						|
            int howmany = m_owner->GetColumnCount();
 | 
						|
            for(int i = m_text.GetCount(); i < howmany; ++i)
 | 
						|
                m_text.Add(wxEmptyString);
 | 
						|
            m_text[col] = text;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    void SetImage(int image, wxTreeItemIcon which) { m_images[which] = image; }
 | 
						|
    void SetImage(size_t col, int image, wxTreeItemIcon which)
 | 
						|
    {
 | 
						|
        if(col == m_owner->GetMainColumn()) m_images[which] = image;
 | 
						|
        else if(col < m_col_images.GetCount())
 | 
						|
            m_col_images[col] = image;
 | 
						|
        else if(col < m_owner->GetColumnCount()) {
 | 
						|
            int howmany = m_owner->GetColumnCount();
 | 
						|
            for(int i = m_col_images.GetCount(); i < howmany; ++i)
 | 
						|
                m_col_images.Add(NO_IMAGE);
 | 
						|
            m_col_images[col] = image;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    void SetData(wxTreeItemData *data) { m_data = data; }
 | 
						|
 | 
						|
    void SetHasPlus(bool has = TRUE) { m_hasPlus = has; }
 | 
						|
 | 
						|
    void SetBold(bool bold) { m_isBold = bold; }
 | 
						|
 | 
						|
    int GetX() const { return m_x; }
 | 
						|
    int GetY() const { return m_y; }
 | 
						|
 | 
						|
    void SetX(int x) { m_x = x; }
 | 
						|
    void SetY(int y) { m_y = y; }
 | 
						|
 | 
						|
    int  GetHeight() const { return m_height; }
 | 
						|
    int  GetWidth()  const { return m_width; }
 | 
						|
 | 
						|
    void SetHeight(int h) { m_height = h; }
 | 
						|
    void SetWidth(int w) { m_width = w; }
 | 
						|
 | 
						|
    wxTreeListItem *GetParent() const { return m_parent; }
 | 
						|
 | 
						|
    // operations
 | 
						|
        // deletes all children notifying the treectrl about it if !NULL
 | 
						|
        // pointer given
 | 
						|
    void DeleteChildren(wxTreeListMainWindow *tree = NULL);
 | 
						|
 | 
						|
    // get count of all children (and grand children if 'recursively')
 | 
						|
    size_t GetChildrenCount(bool recursively = TRUE) const;
 | 
						|
 | 
						|
    void Insert(wxTreeListItem *child, size_t index)
 | 
						|
    { m_children.Insert(child, index); }
 | 
						|
 | 
						|
    void GetSize( int &x, int &y, const wxTreeListMainWindow* );
 | 
						|
 | 
						|
        // return the item at given position (or NULL if no item), onButton is
 | 
						|
        // TRUE if the point belongs to the item's button, otherwise it lies
 | 
						|
        // on the button's label
 | 
						|
    wxTreeListItem *HitTest( const wxPoint& point,
 | 
						|
                             const wxTreeListMainWindow *,
 | 
						|
                             int &flags,
 | 
						|
                             int level );
 | 
						|
    wxTreeListItem *HitTest( const wxPoint& point,
 | 
						|
                             const wxTreeListMainWindow *,
 | 
						|
                             int &flags, int& column /*ALB*/,
 | 
						|
                             int level );
 | 
						|
 | 
						|
    void Expand() { m_isCollapsed = FALSE; }
 | 
						|
    void Collapse() { m_isCollapsed = TRUE; }
 | 
						|
 | 
						|
    void SetHilight( bool set = TRUE ) { m_hasHilight = set; }
 | 
						|
 | 
						|
    // status inquiries
 | 
						|
    bool HasChildren() const { return !m_children.IsEmpty(); }
 | 
						|
    bool IsSelected()  const { return m_hasHilight != 0; }
 | 
						|
    bool IsExpanded()  const { return !m_isCollapsed; }
 | 
						|
    bool HasPlus()     const { return m_hasPlus || HasChildren(); }
 | 
						|
    bool IsBold()      const { return m_isBold != 0; }
 | 
						|
 | 
						|
    // attributes
 | 
						|
        // get them - may be NULL
 | 
						|
    wxTreeItemAttr *GetAttributes() const { return m_attr; }
 | 
						|
        // get them ensuring that the pointer is not NULL
 | 
						|
    wxTreeItemAttr& Attr()
 | 
						|
    {
 | 
						|
        if ( !m_attr )
 | 
						|
        {
 | 
						|
            m_attr = new wxTreeItemAttr;
 | 
						|
            m_ownsAttr = TRUE;
 | 
						|
        }
 | 
						|
        return *m_attr;
 | 
						|
    }
 | 
						|
        // set them
 | 
						|
    void SetAttributes(wxTreeItemAttr *attr)
 | 
						|
    {
 | 
						|
        if ( m_ownsAttr ) delete m_attr;
 | 
						|
        m_attr = attr;
 | 
						|
        m_ownsAttr = FALSE;
 | 
						|
    }
 | 
						|
        // set them and delete when done
 | 
						|
    void AssignAttributes(wxTreeItemAttr *attr)
 | 
						|
    {
 | 
						|
        SetAttributes(attr);
 | 
						|
        m_ownsAttr = TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
private:
 | 
						|
    wxTreeListMainWindow  *m_owner;        // control the item belongs to
 | 
						|
 | 
						|
    // since there can be very many of these, we save size by chosing
 | 
						|
    // the smallest representation for the elements and by ordering
 | 
						|
    // the members to avoid padding.
 | 
						|
    wxArrayString      m_text;    // labels to be rendered for item
 | 
						|
 | 
						|
    wxTreeItemData     *m_data;         // user-provided data
 | 
						|
 | 
						|
    wxArrayTreeListItems m_children; // list of children
 | 
						|
    wxTreeListItem  *m_parent;       // parent of this item
 | 
						|
 | 
						|
    wxTreeItemAttr     *m_attr;         // attributes???
 | 
						|
 | 
						|
    // tree ctrl images for the normal, selected, expanded and
 | 
						|
    // expanded+selected states
 | 
						|
    short               m_images[wxTreeItemIcon_Max];
 | 
						|
    wxArrayShort m_col_images; // images for the various columns (!= main)
 | 
						|
 | 
						|
    wxCoord             m_x;            // (virtual) offset from top
 | 
						|
    wxCoord             m_y;            // (virtual) offset from left
 | 
						|
    short               m_width;        // width of this item
 | 
						|
    unsigned char       m_height;       // height of this item
 | 
						|
 | 
						|
    // use bitfields to save size
 | 
						|
    int                 m_isCollapsed :1;
 | 
						|
    int                 m_hasHilight  :1; // same as focused
 | 
						|
    int                 m_hasPlus     :1; // used for item which doesn't have
 | 
						|
                                          // children but has a [+] button
 | 
						|
    int                 m_isBold      :1; // render the label in bold font
 | 
						|
    int                 m_ownsAttr    :1; // delete attribute when done
 | 
						|
};
 | 
						|
 | 
						|
// ===========================================================================
 | 
						|
// implementation
 | 
						|
// ===========================================================================
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// private functions
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
// translate the key or mouse event flags to the type of selection we're
 | 
						|
// dealing with
 | 
						|
static void EventFlagsToSelType(long style,
 | 
						|
                                bool shiftDown,
 | 
						|
                                bool ctrlDown,
 | 
						|
                                bool &is_multiple,
 | 
						|
                                bool &extended_select,
 | 
						|
                                bool &unselect_others)
 | 
						|
{
 | 
						|
    is_multiple = (style & wxTR_MULTIPLE) != 0;
 | 
						|
    extended_select = shiftDown && is_multiple;
 | 
						|
    unselect_others = !(extended_select || (ctrlDown && is_multiple));
 | 
						|
}
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// wxTreeListRenameTimer (internal)
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
 | 
						|
wxTreeListRenameTimer::wxTreeListRenameTimer( wxTreeListMainWindow *owner )
 | 
						|
{
 | 
						|
    m_owner = owner;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListRenameTimer::Notify()
 | 
						|
{
 | 
						|
    m_owner->OnRenameTimer();
 | 
						|
}
 | 
						|
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
// wxTreeListTextCtrl (internal)
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
 | 
						|
BEGIN_EVENT_TABLE(wxTreeListTextCtrl,wxTextCtrl)
 | 
						|
    EVT_CHAR           (wxTreeListTextCtrl::OnChar)
 | 
						|
    EVT_KEY_UP         (wxTreeListTextCtrl::OnKeyUp)
 | 
						|
    EVT_KILL_FOCUS     (wxTreeListTextCtrl::OnKillFocus)
 | 
						|
END_EVENT_TABLE()
 | 
						|
 | 
						|
wxTreeListTextCtrl::wxTreeListTextCtrl( wxWindow *parent,
 | 
						|
                                        const wxWindowID id,
 | 
						|
                                        bool *accept,
 | 
						|
                                        wxString *res,
 | 
						|
                                        wxTreeListMainWindow *owner,
 | 
						|
                                        const wxString &value,
 | 
						|
                                        const wxPoint &pos,
 | 
						|
                                        const wxSize &size,
 | 
						|
                                        int style,
 | 
						|
                                        const wxValidator& validator,
 | 
						|
                                        const wxString &name )
 | 
						|
    : wxTextCtrl( parent, id, value, pos, size, style, validator, name )
 | 
						|
{
 | 
						|
    m_res = res;
 | 
						|
    m_accept = accept;
 | 
						|
    m_owner = owner;
 | 
						|
    (*m_accept) = FALSE;
 | 
						|
    (*m_res) = wxEmptyString;
 | 
						|
    m_startValue = value;
 | 
						|
    m_finished = FALSE;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListTextCtrl::OnChar( wxKeyEvent &event )
 | 
						|
{
 | 
						|
    if (event.m_keyCode == WXK_RETURN)
 | 
						|
    {
 | 
						|
        (*m_accept) = TRUE;
 | 
						|
        (*m_res) = GetValue();
 | 
						|
 | 
						|
        if ((*m_res) != m_startValue)
 | 
						|
            m_owner->OnRenameAccept();
 | 
						|
 | 
						|
        if (!wxPendingDelete.Member(this))
 | 
						|
            wxPendingDelete.Append(this);
 | 
						|
 | 
						|
        m_finished = TRUE;
 | 
						|
        m_owner->SetFocus(); // This doesn't work. TODO.
 | 
						|
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    if (event.m_keyCode == WXK_ESCAPE)
 | 
						|
    {
 | 
						|
        (*m_accept) = FALSE;
 | 
						|
        (*m_res) = wxEmptyString;
 | 
						|
 | 
						|
        if (!wxPendingDelete.Member(this))
 | 
						|
            wxPendingDelete.Append(this);
 | 
						|
 | 
						|
        m_finished = TRUE;
 | 
						|
        m_owner->SetFocus(); // This doesn't work. TODO.
 | 
						|
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    event.Skip();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListTextCtrl::OnKeyUp( wxKeyEvent &event )
 | 
						|
{
 | 
						|
    if (m_finished)
 | 
						|
    {
 | 
						|
        event.Skip();
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // auto-grow the textctrl:
 | 
						|
    wxSize parentSize = m_owner->GetSize();
 | 
						|
    wxPoint myPos = GetPosition();
 | 
						|
    wxSize mySize = GetSize();
 | 
						|
    int sx, sy;
 | 
						|
    GetTextExtent(GetValue() + _T("M"), &sx, &sy);
 | 
						|
    if (myPos.x + sx > parentSize.x) sx = parentSize.x - myPos.x;
 | 
						|
    if (mySize.x > sx) sx = mySize.x;
 | 
						|
    SetSize(sx, -1);
 | 
						|
 | 
						|
    event.Skip();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListTextCtrl::OnKillFocus( wxFocusEvent &event )
 | 
						|
{
 | 
						|
    if (m_finished)
 | 
						|
    {
 | 
						|
        event.Skip();
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!wxPendingDelete.Member(this))
 | 
						|
        wxPendingDelete.Append(this);
 | 
						|
 | 
						|
    (*m_accept) = TRUE;
 | 
						|
    (*m_res) = GetValue();
 | 
						|
 | 
						|
    if ((*m_res) != m_startValue)
 | 
						|
        m_owner->OnRenameAccept();
 | 
						|
}
 | 
						|
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
//  wxTreeListHeaderWindow
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
 | 
						|
IMPLEMENT_DYNAMIC_CLASS(wxTreeListHeaderWindow,wxWindow);
 | 
						|
 | 
						|
BEGIN_EVENT_TABLE(wxTreeListHeaderWindow,wxWindow)
 | 
						|
    EVT_PAINT         (wxTreeListHeaderWindow::OnPaint)
 | 
						|
    EVT_MOUSE_EVENTS  (wxTreeListHeaderWindow::OnMouse)
 | 
						|
    EVT_SET_FOCUS     (wxTreeListHeaderWindow::OnSetFocus)
 | 
						|
END_EVENT_TABLE()
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::Init()
 | 
						|
{
 | 
						|
    m_currentCursor = (wxCursor *) NULL;
 | 
						|
    m_isDragging = FALSE;
 | 
						|
    m_dirty = FALSE;
 | 
						|
    m_total_col_width = 0;
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListHeaderWindow::wxTreeListHeaderWindow()
 | 
						|
{
 | 
						|
    Init();
 | 
						|
 | 
						|
    m_owner = (wxTreeListMainWindow *) NULL;
 | 
						|
    m_resizeCursor = (wxCursor *) NULL;
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListHeaderWindow::wxTreeListHeaderWindow( wxWindow *win,
 | 
						|
                                                wxWindowID id,
 | 
						|
                                                wxTreeListMainWindow *owner,
 | 
						|
                                                const wxPoint& pos,
 | 
						|
                                                const wxSize& size,
 | 
						|
                                                long style,
 | 
						|
                                                const wxString &name )
 | 
						|
    : wxWindow( win, id, pos, size, style, name )
 | 
						|
{
 | 
						|
    Init();
 | 
						|
 | 
						|
    m_owner = owner;
 | 
						|
    m_resizeCursor = new wxCursor(wxCURSOR_SIZEWE);
 | 
						|
 | 
						|
    SetBackgroundColour(wxSystemSettings::GetSystemColour(
 | 
						|
                            wxSYS_COLOUR_BTNFACE));
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListHeaderWindow::~wxTreeListHeaderWindow()
 | 
						|
{
 | 
						|
    delete m_resizeCursor;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::DoDrawRect( wxDC *dc, int x, int y, int w, int h )
 | 
						|
{
 | 
						|
#ifdef __WXGTK__
 | 
						|
    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);
 | 
						|
#elif defined( __WXMAC__  )
 | 
						|
    const int m_corner = 1;
 | 
						|
 | 
						|
    dc->SetBrush( *wxTRANSPARENT_BRUSH );
 | 
						|
 | 
						|
    dc->SetPen( wxPen(wxSystemSettings::GetSystemColour(
 | 
						|
                          wxSYS_COLOUR_BTNSHADOW), 1, wxSOLID));
 | 
						|
    dc->DrawLine( x+w-m_corner+1, y, x+w, y+h );  // right (outer)
 | 
						|
    dc->DrawRectangle( x, y+h, w+1, 1 );          // bottom (outer)
 | 
						|
 | 
						|
    wxPen pen( wxColour( 0x88 , 0x88 , 0x88 ), 1, wxSOLID );
 | 
						|
 | 
						|
    dc->SetPen( pen );
 | 
						|
    dc->DrawLine( x+w-m_corner, y, x+w-1, y+h );  // right (inner)
 | 
						|
    dc->DrawRectangle( x+1, y+h-1, w-2, 1 );      // bottom (inner)
 | 
						|
 | 
						|
    dc->SetPen( *wxWHITE_PEN );
 | 
						|
    dc->DrawRectangle( x, y, w-m_corner+1, 1 );   // top (outer)
 | 
						|
    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 // !GTK, !Mac
 | 
						|
    const int m_corner = 1;
 | 
						|
 | 
						|
    dc->SetBrush( *wxTRANSPARENT_BRUSH );
 | 
						|
 | 
						|
    dc->SetPen( *wxBLACK_PEN );
 | 
						|
    dc->DrawLine( x+w-m_corner+1, y, x+w, y+h );  // right (outer)
 | 
						|
    dc->DrawRectangle( x, y+h, w+1, 1 );          // bottom (outer)
 | 
						|
 | 
						|
    wxPen pen(wxSystemSettings::GetSystemColour(
 | 
						|
                  wxSYS_COLOUR_BTNSHADOW ), 1, wxSOLID);
 | 
						|
 | 
						|
    dc->SetPen( pen );
 | 
						|
    dc->DrawLine( x+w-m_corner, y, x+w-1, y+h );  // right (inner)
 | 
						|
    dc->DrawRectangle( x+1, y+h-1, w-2, 1 );      // bottom (inner)
 | 
						|
 | 
						|
    dc->SetPen( *wxWHITE_PEN );
 | 
						|
    dc->DrawRectangle( x, y, w-m_corner+1, 1 );   // top (outer)
 | 
						|
    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 );
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
// shift the DC origin to match the position of the main window horz
 | 
						|
// scrollbar: this allows us to always use logical coords
 | 
						|
void wxTreeListHeaderWindow::AdjustDC(wxDC& dc)
 | 
						|
{
 | 
						|
    int xpix;
 | 
						|
    m_owner->GetScrollPixelsPerUnit( &xpix, NULL );
 | 
						|
 | 
						|
    int x;
 | 
						|
    m_owner->GetViewStart( &x, NULL );
 | 
						|
 | 
						|
    // account for the horz scrollbar offset
 | 
						|
    dc.SetDeviceOrigin( -x * xpix, 0 );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
 | 
						|
{
 | 
						|
    static const int HEADER_OFFSET_X = 1, HEADER_OFFSET_Y = 1;
 | 
						|
#ifdef __WXGTK__
 | 
						|
    wxClientDC dc( this );
 | 
						|
#else
 | 
						|
    wxPaintDC dc( this );
 | 
						|
#endif
 | 
						|
 | 
						|
    PrepareDC( dc );
 | 
						|
    AdjustDC( dc );
 | 
						|
 | 
						|
    dc.BeginDrawing();
 | 
						|
 | 
						|
    dc.SetFont( GetFont() );
 | 
						|
 | 
						|
    // width and height of the entire header window
 | 
						|
    int w, h;
 | 
						|
    GetClientSize( &w, &h );
 | 
						|
    m_owner->CalcUnscrolledPosition(w, 0, &w, NULL);
 | 
						|
 | 
						|
    dc.SetBackgroundMode(wxTRANSPARENT);
 | 
						|
 | 
						|
    // do *not* use the listctrl colour for headers - one day we will have a
 | 
						|
    // function to set it separately
 | 
						|
    //dc.SetTextForeground( *wxBLACK );
 | 
						|
    dc.SetTextForeground(wxSystemSettings::
 | 
						|
                            GetSystemColour( wxSYS_COLOUR_WINDOWTEXT ));
 | 
						|
 | 
						|
    int x = HEADER_OFFSET_X;
 | 
						|
 | 
						|
    int numColumns = GetColumnCount();
 | 
						|
    for ( int i = 0; i < numColumns && x < w; i++ )
 | 
						|
    {
 | 
						|
        wxTreeListColumnInfo& column = GetColumn(i);
 | 
						|
        int wCol = column.GetWidth();
 | 
						|
 | 
						|
        // the width of the rect to draw: make it smaller to fit entirely
 | 
						|
        // inside the column rect
 | 
						|
        int cw = wCol - 2;
 | 
						|
 | 
						|
        dc.SetPen( *wxWHITE_PEN );
 | 
						|
 | 
						|
        DoDrawRect( &dc, x, HEADER_OFFSET_Y, cw, h-2 );
 | 
						|
 | 
						|
        // if we have an image, draw it on the right of the label
 | 
						|
        int image = column.GetImage(); //item.m_image;
 | 
						|
        int ix = -2, iy = 0;
 | 
						|
        wxImageList* imageList = m_owner->GetImageList();
 | 
						|
        if(image != -1) {
 | 
						|
            if(imageList) {
 | 
						|
                imageList->GetSize(image, ix, iy);
 | 
						|
            }
 | 
						|
            //else: ignore the column image
 | 
						|
        }
 | 
						|
 | 
						|
        // extra margins around the text label
 | 
						|
        static const int EXTRA_WIDTH = 3;
 | 
						|
        static const int EXTRA_HEIGHT = 4;
 | 
						|
 | 
						|
        int text_width = 0;
 | 
						|
        int text_x = x;
 | 
						|
        int image_offset = cw - ix - 1;
 | 
						|
 | 
						|
        switch(column.GetAlignment()) {
 | 
						|
        case wxTL_ALIGN_LEFT:
 | 
						|
            text_x += EXTRA_WIDTH;
 | 
						|
            cw -= ix + 2;
 | 
						|
            break;
 | 
						|
        case wxTL_ALIGN_RIGHT:
 | 
						|
            dc.GetTextExtent(column.GetText(), &text_width, NULL);
 | 
						|
            text_x += cw - text_width - EXTRA_WIDTH;
 | 
						|
            image_offset = 0;
 | 
						|
            break;
 | 
						|
        case wxTL_ALIGN_CENTER:
 | 
						|
            dc.GetTextExtent(column.GetText(), &text_width, NULL);
 | 
						|
            text_x += (cw - text_width)/2 + ix + 2;
 | 
						|
            image_offset = (cw - text_width - ix - 2)/2;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        // draw the image
 | 
						|
        if(image != -1 && imageList) {
 | 
						|
            imageList->Draw(image, dc, x + image_offset/*cw - ix - 1*/,
 | 
						|
                            HEADER_OFFSET_Y + (h - 4 - iy)/2,
 | 
						|
                            wxIMAGELIST_DRAW_TRANSPARENT);
 | 
						|
        }
 | 
						|
 | 
						|
        // draw the text clipping it so that it doesn't overwrite the column
 | 
						|
        // boundary
 | 
						|
        wxDCClipper clipper(dc, x, HEADER_OFFSET_Y, cw, h - 4 );
 | 
						|
 | 
						|
        dc.DrawText( column.GetText(),
 | 
						|
                     text_x, HEADER_OFFSET_Y + EXTRA_HEIGHT );
 | 
						|
 | 
						|
        x += wCol;
 | 
						|
    }
 | 
						|
 | 
						|
    dc.EndDrawing();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::DrawCurrent()
 | 
						|
{
 | 
						|
    int x1 = m_currentX;
 | 
						|
    int y1 = 0;
 | 
						|
    ClientToScreen( &x1, &y1 );
 | 
						|
 | 
						|
    int x2 = m_currentX-1;
 | 
						|
#ifdef __WXMSW__
 | 
						|
    ++x2; // but why ?
 | 
						|
#endif
 | 
						|
    int y2 = 0;
 | 
						|
    m_owner->GetClientSize( NULL, &y2 );
 | 
						|
    m_owner->ClientToScreen( &x2, &y2 );
 | 
						|
 | 
						|
    wxScreenDC dc;
 | 
						|
    dc.SetLogicalFunction( wxINVERT );
 | 
						|
    dc.SetPen( wxPen( *wxBLACK, 2, wxSOLID ) );
 | 
						|
    dc.SetBrush( *wxTRANSPARENT_BRUSH );
 | 
						|
 | 
						|
    AdjustDC(dc);
 | 
						|
 | 
						|
    dc.DrawLine( x1, y1, x2, y2 );
 | 
						|
 | 
						|
    dc.SetLogicalFunction( wxCOPY );
 | 
						|
 | 
						|
    dc.SetPen( wxNullPen );
 | 
						|
    dc.SetBrush( wxNullBrush );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::OnMouse( wxMouseEvent &event )
 | 
						|
{
 | 
						|
    // we want to work with logical coords
 | 
						|
    int x;
 | 
						|
    m_owner->CalcUnscrolledPosition(event.GetX(), 0, &x, NULL);
 | 
						|
    int y = event.GetY();
 | 
						|
 | 
						|
    if (m_isDragging)
 | 
						|
    {
 | 
						|
        SendListEvent(wxEVT_COMMAND_LIST_COL_DRAGGING,
 | 
						|
                      event.GetPosition());
 | 
						|
 | 
						|
        // we don't draw the line beyond our window, but we allow dragging it
 | 
						|
        // there
 | 
						|
        int w = 0;
 | 
						|
        GetClientSize( &w, NULL );
 | 
						|
        m_owner->CalcUnscrolledPosition(w, 0, &w, NULL);
 | 
						|
        w -= 6;
 | 
						|
 | 
						|
        // erase the line if it was drawn
 | 
						|
        if ( m_currentX < w )
 | 
						|
            DrawCurrent();
 | 
						|
 | 
						|
        if (event.ButtonUp())
 | 
						|
        {
 | 
						|
            ReleaseMouse();
 | 
						|
            m_isDragging = FALSE;
 | 
						|
            m_dirty = TRUE;
 | 
						|
            SetColumnWidth( m_column, m_currentX - m_minX );
 | 
						|
            Refresh();
 | 
						|
            SendListEvent(wxEVT_COMMAND_LIST_COL_END_DRAG,
 | 
						|
                          event.GetPosition());
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            if (x > m_minX + 7)
 | 
						|
                m_currentX = x;
 | 
						|
            else
 | 
						|
                m_currentX = m_minX + 7;
 | 
						|
 | 
						|
            // draw in the new location
 | 
						|
            if ( m_currentX < w )
 | 
						|
                DrawCurrent();
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else // not dragging
 | 
						|
    {
 | 
						|
        m_minX = 0;
 | 
						|
        bool hit_border = FALSE;
 | 
						|
 | 
						|
        // end of the current column
 | 
						|
        int xpos = 0;
 | 
						|
 | 
						|
        // find the column where this event occured
 | 
						|
        int countCol = GetColumnCount();
 | 
						|
        for (int col = 0; col < countCol; col++)
 | 
						|
        {
 | 
						|
            xpos += GetColumnWidth( col );
 | 
						|
            m_column = col;
 | 
						|
 | 
						|
            if ( (abs(x-xpos) < 3) && (y < 22) )
 | 
						|
            {
 | 
						|
                // near the column border
 | 
						|
                hit_border = TRUE;
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            if ( x < xpos )
 | 
						|
            {
 | 
						|
                // inside the column
 | 
						|
                break;
 | 
						|
            }
 | 
						|
 | 
						|
            m_minX = xpos;
 | 
						|
        }
 | 
						|
 | 
						|
        if (event.LeftDown() || event.RightUp())
 | 
						|
        {
 | 
						|
            if (hit_border && event.LeftDown())
 | 
						|
            {
 | 
						|
                m_isDragging = TRUE;
 | 
						|
                m_currentX = x;
 | 
						|
                DrawCurrent();
 | 
						|
                CaptureMouse();
 | 
						|
                SendListEvent(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG,
 | 
						|
                              event.GetPosition());
 | 
						|
            }
 | 
						|
            else // click on a column
 | 
						|
            {
 | 
						|
                SendListEvent( event.LeftDown()
 | 
						|
                               ? wxEVT_COMMAND_LIST_COL_CLICK
 | 
						|
                               : wxEVT_COMMAND_LIST_COL_RIGHT_CLICK,
 | 
						|
                               event.GetPosition());
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else if (event.Moving())
 | 
						|
        {
 | 
						|
            bool setCursor;
 | 
						|
            if (hit_border)
 | 
						|
            {
 | 
						|
                setCursor = m_currentCursor == wxSTANDARD_CURSOR;
 | 
						|
                m_currentCursor = m_resizeCursor;
 | 
						|
            }
 | 
						|
            else
 | 
						|
            {
 | 
						|
                setCursor = m_currentCursor != wxSTANDARD_CURSOR;
 | 
						|
                m_currentCursor = wxSTANDARD_CURSOR;
 | 
						|
            }
 | 
						|
 | 
						|
            if ( setCursor )
 | 
						|
                SetCursor(*m_currentCursor);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::OnSetFocus( wxFocusEvent &WXUNUSED(event) )
 | 
						|
{
 | 
						|
    m_owner->SetFocus();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListHeaderWindow::SendListEvent(wxEventType type, wxPoint pos)
 | 
						|
{
 | 
						|
    wxWindow *parent = GetParent();
 | 
						|
    wxListEvent le( type, parent->GetId() );
 | 
						|
    le.SetEventObject( parent );
 | 
						|
    le.m_pointDrag = pos;
 | 
						|
 | 
						|
    // the position should be relative to the parent window, not
 | 
						|
    // this one for compatibility with MSW and common sense: the
 | 
						|
    // user code doesn't know anything at all about this header
 | 
						|
    // window, so why should it get positions relative to it?
 | 
						|
    le.m_pointDrag.y -= GetSize().y;
 | 
						|
 | 
						|
    le.m_col = m_column;
 | 
						|
    parent->GetEventHandler()->ProcessEvent( le );
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListHeaderWindow::AddColumn(const wxTreeListColumnInfo& col)
 | 
						|
{
 | 
						|
    m_columns.Add(col);
 | 
						|
    m_total_col_width += col.GetWidth();
 | 
						|
    //m_owner->GetHeaderWindow()->Refresh();
 | 
						|
    //m_dirty = TRUE;
 | 
						|
    m_owner->AdjustMyScrollbars();
 | 
						|
    m_owner->m_dirty = TRUE;
 | 
						|
    Refresh();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListHeaderWindow::SetColumnWidth(size_t column, size_t width)
 | 
						|
{
 | 
						|
    if(column < GetColumnCount()) {
 | 
						|
        m_total_col_width -= m_columns[column].GetWidth();
 | 
						|
        m_columns[column].SetWidth(width);
 | 
						|
        m_total_col_width += width;
 | 
						|
        m_owner->AdjustMyScrollbars();
 | 
						|
        m_owner->m_dirty = TRUE;
 | 
						|
        //m_dirty = TRUE;
 | 
						|
        Refresh();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListHeaderWindow::InsertColumn(size_t before,
 | 
						|
                                          const wxTreeListColumnInfo& col)
 | 
						|
{
 | 
						|
    wxCHECK_RET(before < GetColumnCount(), wxT("Invalid column index"));
 | 
						|
    m_columns.Insert(col, before);
 | 
						|
    m_total_col_width += col.GetWidth();
 | 
						|
    //m_dirty = TRUE;
 | 
						|
    //m_owner->GetHeaderWindow()->Refresh();
 | 
						|
    m_owner->AdjustMyScrollbars();
 | 
						|
    m_owner->m_dirty = TRUE;
 | 
						|
    Refresh();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListHeaderWindow::RemoveColumn(size_t column)
 | 
						|
{
 | 
						|
    wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
 | 
						|
    m_total_col_width -= m_columns[column].GetWidth();
 | 
						|
    m_columns.RemoveAt(column);
 | 
						|
    //m_dirty = TRUE;
 | 
						|
    m_owner->AdjustMyScrollbars();
 | 
						|
    m_owner->m_dirty = TRUE;
 | 
						|
    Refresh();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListHeaderWindow::SetColumn(size_t column,
 | 
						|
                                       const wxTreeListColumnInfo& info)
 | 
						|
{
 | 
						|
    wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
 | 
						|
    size_t w = m_columns[column].GetWidth();
 | 
						|
    m_columns[column] = info;
 | 
						|
    //m_owner->GetHeaderWindow()->Refresh();
 | 
						|
    //m_dirty = TRUE;
 | 
						|
    if(w != info.GetWidth()) {
 | 
						|
        m_total_col_width += info.GetWidth() - w;
 | 
						|
        m_owner->AdjustMyScrollbars();
 | 
						|
        m_owner->m_dirty = TRUE;
 | 
						|
    }
 | 
						|
    Refresh();
 | 
						|
}
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// wxTreeListItem
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
 | 
						|
wxTreeListItem::wxTreeListItem(wxTreeListMainWindow *owner,
 | 
						|
                                     wxTreeListItem *parent,
 | 
						|
                                     const wxArrayString& text,
 | 
						|
                                     int image, int selImage,
 | 
						|
                                     wxTreeItemData *data)
 | 
						|
                 : m_text(text)
 | 
						|
{
 | 
						|
    m_images[wxTreeItemIcon_Normal] = image;
 | 
						|
    m_images[wxTreeItemIcon_Selected] = selImage;
 | 
						|
    m_images[wxTreeItemIcon_Expanded] = NO_IMAGE;
 | 
						|
    m_images[wxTreeItemIcon_SelectedExpanded] = NO_IMAGE;
 | 
						|
 | 
						|
    m_data = data;
 | 
						|
    m_x = m_y = 0;
 | 
						|
 | 
						|
    m_isCollapsed = TRUE;
 | 
						|
    m_hasHilight = FALSE;
 | 
						|
    m_hasPlus = FALSE;
 | 
						|
    m_isBold = FALSE;
 | 
						|
 | 
						|
    m_owner = owner;
 | 
						|
 | 
						|
    m_parent = parent;
 | 
						|
 | 
						|
    m_attr = (wxTreeItemAttr *)NULL;
 | 
						|
    m_ownsAttr = FALSE;
 | 
						|
 | 
						|
    // We don't know the height here yet.
 | 
						|
    m_width = 0;
 | 
						|
    m_height = 0;
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListItem::~wxTreeListItem()
 | 
						|
{
 | 
						|
    delete m_data;
 | 
						|
 | 
						|
    if (m_ownsAttr) delete m_attr;
 | 
						|
 | 
						|
    wxASSERT_MSG( m_children.IsEmpty(),
 | 
						|
                  wxT("please call DeleteChildren() before deleting the item") );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListItem::DeleteChildren(wxTreeListMainWindow *tree)
 | 
						|
{
 | 
						|
    size_t count = m_children.Count();
 | 
						|
    for ( size_t n = 0; n < count; n++ )
 | 
						|
    {
 | 
						|
        wxTreeListItem *child = m_children[n];
 | 
						|
        if (tree)
 | 
						|
            tree->SendDeleteEvent(child);
 | 
						|
 | 
						|
        child->DeleteChildren(tree);
 | 
						|
        delete child;
 | 
						|
    }
 | 
						|
 | 
						|
    m_children.Empty();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListItem::SetText( const wxString &text )
 | 
						|
{
 | 
						|
    if(m_text.GetCount() > 0) m_text[0] = text;
 | 
						|
    else {
 | 
						|
        m_text.Add(text);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
size_t wxTreeListItem::GetChildrenCount(bool recursively) const
 | 
						|
{
 | 
						|
    size_t count = m_children.Count();
 | 
						|
    if ( !recursively )
 | 
						|
        return count;
 | 
						|
 | 
						|
    size_t total = count;
 | 
						|
    for (size_t n = 0; n < count; ++n)
 | 
						|
    {
 | 
						|
        total += m_children[n]->GetChildrenCount();
 | 
						|
    }
 | 
						|
 | 
						|
    return total;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListItem::GetSize( int &x, int &y,
 | 
						|
                                 const wxTreeListMainWindow *theButton )
 | 
						|
{
 | 
						|
    int bottomY=m_y+theButton->GetLineHeight(this);
 | 
						|
    if ( y < bottomY ) y = bottomY;
 | 
						|
    int width = m_x +  m_width;
 | 
						|
    if ( x < width ) x = width;
 | 
						|
 | 
						|
    if (IsExpanded())
 | 
						|
    {
 | 
						|
        size_t count = m_children.Count();
 | 
						|
        for ( size_t n = 0; n < count; ++n )
 | 
						|
        {
 | 
						|
            m_children[n]->GetSize( x, y, theButton );
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListItem *wxTreeListItem::HitTest(const wxPoint& point,
 | 
						|
                                        const wxTreeListMainWindow *theCtrl,
 | 
						|
                                        int &flags,
 | 
						|
                                        int level)
 | 
						|
{
 | 
						|
    // for a hidden root node, don't evaluate it, but do evaluate children
 | 
						|
    if ( !(level == 0 && theCtrl->HasFlag(wxTR_HIDE_ROOT)) )
 | 
						|
    {
 | 
						|
        // evaluate the item
 | 
						|
        int h = theCtrl->GetLineHeight(this);
 | 
						|
        if ((point.y > m_y) && (point.y <= m_y + h))
 | 
						|
        {
 | 
						|
            int y_mid = m_y + h/2;
 | 
						|
            if (point.y < y_mid )
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMUPPERPART;
 | 
						|
            else
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMLOWERPART;
 | 
						|
 | 
						|
            // 5 is the size of the plus sign
 | 
						|
            int xCross = m_x - theCtrl->GetSpacing();
 | 
						|
            if ((point.x > xCross-5) && (point.x < xCross+5) &&
 | 
						|
                (point.y > y_mid-5) && (point.y < y_mid+5) &&
 | 
						|
                HasPlus() && theCtrl->HasButtons() )
 | 
						|
            {
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMBUTTON;
 | 
						|
                return this;
 | 
						|
            }
 | 
						|
 | 
						|
            if ((point.x >= m_x) && (point.x <= m_x+m_width))
 | 
						|
            {
 | 
						|
                int image_w = -1;
 | 
						|
                int image_h;
 | 
						|
 | 
						|
                //assuming every image (normal and selected) has the same size!
 | 
						|
                if ( (GetImage() != NO_IMAGE) && theCtrl->m_imageListNormal )
 | 
						|
                    theCtrl->m_imageListNormal->GetSize(GetImage(),
 | 
						|
                                                        image_w, image_h);
 | 
						|
 | 
						|
                if ((image_w != -1) && (point.x <= m_x + image_w + 1))
 | 
						|
                    flags |= wxTREE_HITTEST_ONITEMICON;
 | 
						|
                else
 | 
						|
                    flags |= wxTREE_HITTEST_ONITEMLABEL;
 | 
						|
 | 
						|
                return this;
 | 
						|
            }
 | 
						|
 | 
						|
            if (point.x < m_x)
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMINDENT;
 | 
						|
            if (point.x > m_x+m_width)
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMRIGHT;
 | 
						|
 | 
						|
            return this;
 | 
						|
        }
 | 
						|
 | 
						|
        // if children are expanded, fall through to evaluate them
 | 
						|
        if (m_isCollapsed) return (wxTreeListItem*) NULL;
 | 
						|
    }
 | 
						|
 | 
						|
    // evaluate children
 | 
						|
    size_t count = m_children.Count();
 | 
						|
    for ( size_t n = 0; n < count; n++ )
 | 
						|
    {
 | 
						|
        wxTreeListItem *res = m_children[n]->HitTest(point, theCtrl,
 | 
						|
                                                     flags, level + 1);
 | 
						|
        if ( res != NULL )
 | 
						|
            return res;
 | 
						|
    }
 | 
						|
 | 
						|
    return (wxTreeListItem*) NULL;
 | 
						|
}
 | 
						|
 | 
						|
// ALB
 | 
						|
wxTreeListItem *wxTreeListItem::HitTest(const wxPoint& point,
 | 
						|
                                        const wxTreeListMainWindow *theCtrl,
 | 
						|
                                        int &flags, int& column, int level)
 | 
						|
{
 | 
						|
    column = theCtrl->GetMainColumn(); //-1;
 | 
						|
    wxTreeListItem* res = HitTest(point, theCtrl, flags, level);
 | 
						|
 | 
						|
    if (!res) {
 | 
						|
        column = -1;
 | 
						|
        return res;
 | 
						|
    }
 | 
						|
  
 | 
						|
    if (point.x >= theCtrl->m_owner->GetHeaderWindow()->GetWidth())
 | 
						|
        column = -1;
 | 
						|
    else if (flags & wxTREE_HITTEST_ONITEMINDENT) {
 | 
						|
        int x = 0;
 | 
						|
        for (int i = 0; i < column; ++i) {
 | 
						|
            int w = theCtrl->m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
            if(point.x >= x && point.x < x+w) {
 | 
						|
                flags ^= wxTREE_HITTEST_ONITEMINDENT;
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMCOLUMN;
 | 
						|
                column = i;
 | 
						|
                return res;
 | 
						|
            }
 | 
						|
            x += w;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else if (flags & wxTREE_HITTEST_ONITEMRIGHT) {
 | 
						|
        int x = 0;
 | 
						|
        int i;
 | 
						|
        for (i = 0; i < column+1; ++i) {
 | 
						|
            x += theCtrl->m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
        }
 | 
						|
        for (i = column+1; i < (int)theCtrl->GetColumnCount(); ++i) {
 | 
						|
            int w = theCtrl->m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
            if (point.x >= x && point.x < x+w) {
 | 
						|
                flags ^= wxTREE_HITTEST_ONITEMRIGHT;
 | 
						|
                flags |= wxTREE_HITTEST_ONITEMCOLUMN;
 | 
						|
                column = i;
 | 
						|
                return res;
 | 
						|
            }
 | 
						|
            x += w;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    return res;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int wxTreeListItem::GetCurrentImage() const
 | 
						|
{
 | 
						|
    int image = NO_IMAGE;
 | 
						|
    if ( IsExpanded() )
 | 
						|
    {
 | 
						|
        if ( IsSelected() )
 | 
						|
        {
 | 
						|
            image = GetImage(wxTreeItemIcon_SelectedExpanded);
 | 
						|
        }
 | 
						|
 | 
						|
        if ( image == NO_IMAGE )
 | 
						|
        {
 | 
						|
            // we usually fall back to the normal item, but try just the
 | 
						|
            // expanded one (and not selected) first in this case
 | 
						|
            image = GetImage(wxTreeItemIcon_Expanded);
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else // not expanded
 | 
						|
    {
 | 
						|
        if ( IsSelected() )
 | 
						|
            image = GetImage(wxTreeItemIcon_Selected);
 | 
						|
    }
 | 
						|
 | 
						|
    // maybe it doesn't have the specific image we want,
 | 
						|
    // try the default one instead
 | 
						|
    if ( image == NO_IMAGE ) image = GetImage();
 | 
						|
 | 
						|
    return image;
 | 
						|
}
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// wxTreeListMainWindow implementation
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
 | 
						|
IMPLEMENT_DYNAMIC_CLASS(wxTreeListMainWindow, wxScrolledWindow)
 | 
						|
 | 
						|
BEGIN_EVENT_TABLE(wxTreeListMainWindow, wxScrolledWindow)
 | 
						|
    EVT_PAINT          (wxTreeListMainWindow::OnPaint)
 | 
						|
    EVT_MOUSE_EVENTS   (wxTreeListMainWindow::OnMouse)
 | 
						|
    EVT_CHAR           (wxTreeListMainWindow::OnChar)
 | 
						|
    EVT_SET_FOCUS      (wxTreeListMainWindow::OnSetFocus)
 | 
						|
    EVT_KILL_FOCUS     (wxTreeListMainWindow::OnKillFocus)
 | 
						|
    EVT_IDLE           (wxTreeListMainWindow::OnIdle)
 | 
						|
    //EVT_SIZE           (wxTreeListMainWindow::OnSize)
 | 
						|
    EVT_SCROLLWIN      (wxTreeListMainWindow::OnScroll)
 | 
						|
END_EVENT_TABLE()
 | 
						|
 | 
						|
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
// construction/destruction
 | 
						|
// ---------------------------------------------------------------------------
 | 
						|
 | 
						|
void wxTreeListMainWindow::Init()
 | 
						|
{
 | 
						|
    m_current = m_key_current = m_anchor = (wxTreeListItem *) NULL;
 | 
						|
    m_hasFocus = FALSE;
 | 
						|
    m_dirty = FALSE;
 | 
						|
 | 
						|
    m_lineHeight = 10;
 | 
						|
    m_indent = 9;
 | 
						|
    m_spacing = 9;
 | 
						|
    m_linespacing = 4;
 | 
						|
 | 
						|
    m_hilightBrush = new wxBrush
 | 
						|
                         (
 | 
						|
                            wxSystemSettings::GetSystemColour
 | 
						|
                            (
 | 
						|
                                wxSYS_COLOUR_HIGHLIGHT
 | 
						|
                            ),
 | 
						|
                            wxSOLID
 | 
						|
                         );
 | 
						|
 | 
						|
    m_hilightUnfocusedBrush = new wxBrush
 | 
						|
                              (
 | 
						|
                                 wxSystemSettings::GetSystemColour
 | 
						|
                                 (
 | 
						|
                                     wxSYS_COLOUR_BTNSHADOW
 | 
						|
                                 ),
 | 
						|
                                 wxSOLID
 | 
						|
                              );
 | 
						|
 | 
						|
    m_imageListNormal = m_imageListButtons =
 | 
						|
    m_imageListState = (wxImageList *) NULL;
 | 
						|
    m_ownsImageListNormal = m_ownsImageListButtons =
 | 
						|
    m_ownsImageListState = FALSE;
 | 
						|
 | 
						|
    m_dragCount = 0;
 | 
						|
    m_isDragging = FALSE;
 | 
						|
    m_dropTarget = m_oldSelection = (wxTreeListItem *)NULL;
 | 
						|
 | 
						|
    m_renameTimer = new wxTreeListRenameTimer( this );
 | 
						|
    m_lastOnSame = FALSE;
 | 
						|
 | 
						|
    m_normalFont = wxSystemSettings::GetSystemFont( wxSYS_DEFAULT_GUI_FONT );
 | 
						|
    m_boldFont = wxFont( m_normalFont.GetPointSize(),
 | 
						|
                         m_normalFont.GetFamily(),
 | 
						|
                         m_normalFont.GetStyle(),
 | 
						|
                         wxBOLD,
 | 
						|
                         m_normalFont.GetUnderlined());
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
static const int HEADER_HEIGHT = 23;
 | 
						|
 | 
						|
bool wxTreeListMainWindow::Create(wxTreeListCtrl *parent,
 | 
						|
                                  wxWindowID id,
 | 
						|
                                  const wxPoint& pos,
 | 
						|
                                  const wxSize& size,
 | 
						|
                                  long style,
 | 
						|
                                  const wxValidator &validator,
 | 
						|
                                  const wxString& name )
 | 
						|
{
 | 
						|
#ifdef __WXMAC__
 | 
						|
    int major,minor;
 | 
						|
    wxGetOsVersion( &major, &minor );
 | 
						|
 | 
						|
    if (style & wxTR_HAS_BUTTONS) style |= wxTR_MAC_BUTTONS;
 | 
						|
    if (style & wxTR_HAS_BUTTONS) style &= ~wxTR_HAS_BUTTONS;
 | 
						|
    style &= ~wxTR_LINES_AT_ROOT;
 | 
						|
    style |= wxTR_NO_LINES;
 | 
						|
    if (major < 10)
 | 
						|
        style |= wxTR_ROW_LINES;
 | 
						|
#endif
 | 
						|
 | 
						|
    wxScrolledWindow::Create( parent, id, pos, size,
 | 
						|
                              style|wxHSCROLL|wxVSCROLL, name );
 | 
						|
 | 
						|
        // If the tree display has no buttons, but does have
 | 
						|
        // connecting lines, we can use a narrower layout.
 | 
						|
        // It may not be a good idea to force this...
 | 
						|
    if (!HasButtons() && !HasFlag(wxTR_NO_LINES))
 | 
						|
    {
 | 
						|
        m_indent= 10;
 | 
						|
        m_spacing = 10;
 | 
						|
    }
 | 
						|
 | 
						|
#if wxUSE_VALIDATORS
 | 
						|
    SetValidator( validator );
 | 
						|
#endif
 | 
						|
 | 
						|
    SetBackgroundColour( wxSystemSettings::GetSystemColour( wxSYS_COLOUR_LISTBOX ) );
 | 
						|
 | 
						|
// #ifdef __WXMSW__
 | 
						|
//     m_dottedPen = wxPen( "black", 0, wxDOT );  // too slow under XFree86
 | 
						|
// #else
 | 
						|
    m_dottedPen = wxPen( wxT("grey"), 0, 0 );
 | 
						|
// #endif
 | 
						|
 | 
						|
    // ALB
 | 
						|
    m_owner = parent;
 | 
						|
    m_main_column = 0;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListMainWindow::~wxTreeListMainWindow()
 | 
						|
{
 | 
						|
    delete m_hilightBrush;
 | 
						|
    delete m_hilightUnfocusedBrush;
 | 
						|
 | 
						|
    DeleteAllItems();
 | 
						|
 | 
						|
    delete m_renameTimer;
 | 
						|
    if (m_ownsImageListNormal) delete m_imageListNormal;
 | 
						|
    if (m_ownsImageListState) delete m_imageListState;
 | 
						|
    if (m_ownsImageListButtons) delete m_imageListButtons;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
// accessors
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
 | 
						|
inline
 | 
						|
size_t wxTreeListMainWindow::GetCount() const
 | 
						|
{
 | 
						|
    return m_anchor == NULL ? 0u : m_anchor->GetChildrenCount();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetIndent(unsigned int indent)
 | 
						|
{
 | 
						|
    m_indent = indent;
 | 
						|
    m_dirty = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetSpacing(unsigned int spacing)
 | 
						|
{
 | 
						|
    m_spacing = spacing;
 | 
						|
    m_dirty = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetLineSpacing(unsigned int spacing)
 | 
						|
{
 | 
						|
    m_linespacing = spacing;
 | 
						|
    m_dirty = TRUE;
 | 
						|
    CalculateLineHeight();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
size_t wxTreeListMainWindow::GetChildrenCount(const wxTreeItemId& item,
 | 
						|
                                              bool recursively)
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), 0u, wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->GetChildrenCount(recursively);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::SetWindowStyle(const long styles)
 | 
						|
{
 | 
						|
        // right now, just sets the styles.  Eventually, we may
 | 
						|
        // want to update the inherited styles, but right now
 | 
						|
        // none of the parents has updatable styles
 | 
						|
    m_windowStyle = styles;
 | 
						|
    m_dirty = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
// functions to work with tree items
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
 | 
						|
inline
 | 
						|
int wxTreeListMainWindow::GetItemImage(const wxTreeItemId& item, size_t column,
 | 
						|
                                       wxTreeItemIcon which) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), -1, wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->GetImage(column, which);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemData *wxTreeListMainWindow::GetItemData(const wxTreeItemId& item)
 | 
						|
    const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), NULL, wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->GetData();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::GetItemBold(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG(item.IsOk(), FALSE, wxT("invalid tree item"));
 | 
						|
    return ((wxTreeListItem *)item.m_pItem)->IsBold();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxColour wxTreeListMainWindow::GetItemTextColour(const wxTreeItemId& item)
 | 
						|
    const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxNullColour, wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    return pItem->Attr().GetTextColour();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxColour wxTreeListMainWindow::GetItemBackgroundColour(
 | 
						|
    const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxNullColour, wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    return pItem->Attr().GetBackgroundColour();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxFont wxTreeListMainWindow::GetItemFont(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxNullFont, wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    return pItem->Attr().GetFont();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemImage(const wxTreeItemId& item,
 | 
						|
                                        size_t column,
 | 
						|
                                        int image, wxTreeItemIcon which)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    pItem->SetImage(column, image, which);
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    CalculateSize(pItem, dc);
 | 
						|
    RefreshLine(pItem);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemData(const wxTreeItemId& item,
 | 
						|
                                       wxTreeItemData *data)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    ((wxTreeListItem*) item.m_pItem)->SetData(data);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemHasChildren(const wxTreeItemId& item,
 | 
						|
                                              bool has)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    pItem->SetHasPlus(has);
 | 
						|
    RefreshLine(pItem);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemBold(const wxTreeItemId& item, bool bold)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    // avoid redrawing the tree if no real change
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    if ( pItem->IsBold() != bold )
 | 
						|
    {
 | 
						|
        pItem->SetBold(bold);
 | 
						|
        RefreshLine(pItem);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemTextColour(const wxTreeItemId& item,
 | 
						|
                                             const wxColour& col)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    pItem->Attr().SetTextColour(col);
 | 
						|
    RefreshLine(pItem);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemBackgroundColour(const wxTreeItemId& item,
 | 
						|
                                                   const wxColour& col)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    pItem->Attr().SetBackgroundColour(col);
 | 
						|
    RefreshLine(pItem);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemFont(const wxTreeItemId& item,
 | 
						|
                                       const wxFont& font)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    pItem->Attr().SetFont(font);
 | 
						|
    RefreshLine(pItem);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::SetFont( const wxFont &font )
 | 
						|
{
 | 
						|
    wxScrolledWindow::SetFont(font);
 | 
						|
 | 
						|
    m_normalFont = font ;
 | 
						|
    m_boldFont = wxFont( m_normalFont.GetPointSize(),
 | 
						|
                            m_normalFont.GetFamily(),
 | 
						|
                            m_normalFont.GetStyle(),
 | 
						|
                            wxBOLD,
 | 
						|
                            m_normalFont.GetUnderlined());
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// item status inquiries
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::IsVisible(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
 | 
						|
 | 
						|
    // An item is only visible if it's not a descendant of a collapsed item
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    wxTreeListItem* parent = pItem->GetParent();
 | 
						|
    while (parent)
 | 
						|
    {
 | 
						|
        if (!parent->IsExpanded())
 | 
						|
            return FALSE;
 | 
						|
        parent = parent->GetParent();
 | 
						|
    }
 | 
						|
 | 
						|
    int startX, startY;
 | 
						|
    GetViewStart(& startX, & startY);
 | 
						|
 | 
						|
    wxSize clientSize = GetClientSize();
 | 
						|
 | 
						|
    wxRect rect;
 | 
						|
    if (!GetBoundingRect(item, rect))
 | 
						|
        return FALSE;
 | 
						|
    if (rect.GetWidth() == 0 || rect.GetHeight() == 0)
 | 
						|
        return FALSE;
 | 
						|
    if (rect.GetBottom() < 0 || rect.GetTop() > clientSize.y)
 | 
						|
        return FALSE;
 | 
						|
    if (rect.GetRight() < 0 || rect.GetLeft() > clientSize.x)
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::ItemHasChildren(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
 | 
						|
 | 
						|
    // consider that the item does have children if it has the "+" button: it
 | 
						|
    // might not have them (if it had never been expanded yet) but then it
 | 
						|
    // could have them as well and it's better to err on this side rather than
 | 
						|
    // disabling some operations which are restricted to the items with
 | 
						|
    // children for an item which does have them
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->HasPlus();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::IsExpanded(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->IsExpanded();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::IsSelected(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->IsSelected();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
bool wxTreeListMainWindow::IsBold(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->IsBold();
 | 
						|
}
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// navigation
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetParent(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->GetParent();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetFirstChild(const wxTreeItemId& item,
 | 
						|
                                                 long& cookie) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    cookie = 0;
 | 
						|
    return GetNextChild(item, cookie);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetNextChild(const wxTreeItemId& item,
 | 
						|
                                                long& cookie) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxArrayTreeListItems& children = ((wxTreeListItem*)
 | 
						|
                                      item.m_pItem)->GetChildren();
 | 
						|
    if ( (size_t)cookie < children.Count() )
 | 
						|
    {
 | 
						|
        return children.Item((size_t)cookie++);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        // there are no more of them
 | 
						|
        return wxTreeItemId();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetLastChild(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxArrayTreeListItems& children = ((wxTreeListItem*) item.m_pItem)->GetChildren();
 | 
						|
    return (children.IsEmpty() ? wxTreeItemId() : wxTreeItemId(children.Last()));
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetNextSibling(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
 | 
						|
    wxTreeListItem *parent = i->GetParent();
 | 
						|
    if ( parent == NULL )
 | 
						|
    {
 | 
						|
        // root item doesn't have any siblings
 | 
						|
        return wxTreeItemId();
 | 
						|
    }
 | 
						|
 | 
						|
    wxArrayTreeListItems& siblings = parent->GetChildren();
 | 
						|
    int index = siblings.Index(i);
 | 
						|
    wxASSERT( index != wxNOT_FOUND ); // I'm not a child of my parent?
 | 
						|
 | 
						|
    size_t n = (size_t)(index + 1);
 | 
						|
    return n == siblings.Count() ? wxTreeItemId() : wxTreeItemId(siblings[n]);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetPrevSibling(const wxTreeItemId& item)
 | 
						|
    const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
 | 
						|
    wxTreeListItem *parent = i->GetParent();
 | 
						|
    if ( parent == NULL )
 | 
						|
    {
 | 
						|
        // root item doesn't have any siblings
 | 
						|
        return wxTreeItemId();
 | 
						|
    }
 | 
						|
 | 
						|
    wxArrayTreeListItems& siblings = parent->GetChildren();
 | 
						|
    int index = siblings.Index(i);
 | 
						|
    wxASSERT( index != wxNOT_FOUND ); // I'm not a child of my parent?
 | 
						|
 | 
						|
    return index == 0 ? wxTreeItemId()
 | 
						|
                      : wxTreeItemId(siblings[(size_t)(index - 1)]);
 | 
						|
}
 | 
						|
 | 
						|
// Only for internal use right now, but should probably be public
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetNext(const wxTreeItemId& item) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    // First see if there are any children.
 | 
						|
    wxArrayTreeListItems& children = i->GetChildren();
 | 
						|
    if (children.GetCount() > 0)
 | 
						|
    {
 | 
						|
         return children.Item(0);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
         // Try a sibling of this or ancestor instead
 | 
						|
         wxTreeItemId p = item;
 | 
						|
         wxTreeItemId toFind;
 | 
						|
         do
 | 
						|
         {
 | 
						|
              toFind = GetNextSibling(p);
 | 
						|
              p = GetParent(p);
 | 
						|
         } while (p.IsOk() && !toFind.IsOk());
 | 
						|
         return toFind;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetFirstVisibleItem() const
 | 
						|
{
 | 
						|
    wxTreeItemId id = GetRootItem();
 | 
						|
    if (!id.IsOk())
 | 
						|
        return id;
 | 
						|
 | 
						|
    do
 | 
						|
    {
 | 
						|
        if (IsVisible(id))
 | 
						|
              return id;
 | 
						|
        id = GetNext(id);
 | 
						|
    } while (id.IsOk());
 | 
						|
 | 
						|
    return wxTreeItemId();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetNextVisible(const wxTreeItemId& item)
 | 
						|
    const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeItemId id = item;
 | 
						|
    if (id.IsOk())
 | 
						|
    {
 | 
						|
        while (id = GetNext(id), id.IsOk())
 | 
						|
        {
 | 
						|
            if (IsVisible(id))
 | 
						|
                return id;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return wxTreeItemId();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::GetPrevVisible(const wxTreeItemId& item)
 | 
						|
    const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxFAIL_MSG(wxT("not implemented"));
 | 
						|
 | 
						|
    return wxTreeItemId();
 | 
						|
}
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// operations
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
wxTreeItemId wxTreeListMainWindow::DoInsertItem(const wxTreeItemId& parentId,
 | 
						|
                                      size_t previous,
 | 
						|
                                      const wxString& text,
 | 
						|
                                      int image, int selImage,
 | 
						|
                                      wxTreeItemData *data)
 | 
						|
{
 | 
						|
    wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
 | 
						|
    if ( !parent )
 | 
						|
    {
 | 
						|
        // should we give a warning here?
 | 
						|
        return AddRoot(text, image, selImage, data);
 | 
						|
    }
 | 
						|
 | 
						|
    m_dirty = TRUE;     // do this first so stuff below doesn't cause flicker
 | 
						|
 | 
						|
    // ALB
 | 
						|
    wxArrayString arr;
 | 
						|
    arr.Alloc(GetColumnCount());
 | 
						|
    for(size_t i = 0; i < GetColumnCount(); ++i) {
 | 
						|
        arr.Add(wxEmptyString);
 | 
						|
    }
 | 
						|
    arr[m_main_column] = text;
 | 
						|
    wxTreeListItem *item =
 | 
						|
        new wxTreeListItem( this, parent, arr, image, selImage, data );
 | 
						|
 | 
						|
    if ( data != NULL )
 | 
						|
    {
 | 
						|
        data->SetId((long)item);
 | 
						|
    }
 | 
						|
 | 
						|
    parent->Insert( item, previous );
 | 
						|
 | 
						|
    return item;
 | 
						|
}
 | 
						|
 | 
						|
wxTreeItemId wxTreeListMainWindow::AddRoot(const wxString& text,
 | 
						|
                                 int image, int selImage,
 | 
						|
                                 wxTreeItemData *data)
 | 
						|
{
 | 
						|
    wxCHECK_MSG(!m_anchor, wxTreeItemId(), wxT("tree can have only one root"));
 | 
						|
    wxCHECK_MSG(GetColumnCount(), wxTreeItemId(), wxT("Add column(s) before adding the root item"));
 | 
						|
 | 
						|
    m_dirty = TRUE;     // do this first so stuff below doesn't cause flicker
 | 
						|
 | 
						|
    // ALB
 | 
						|
    wxArrayString arr;
 | 
						|
    arr.Alloc(GetColumnCount());
 | 
						|
    for(size_t i = 0; i < GetColumnCount(); ++i) {
 | 
						|
        arr.Add(wxEmptyString);
 | 
						|
    }
 | 
						|
    arr[m_main_column] = text;
 | 
						|
    m_anchor = new wxTreeListItem( this, (wxTreeListItem *)NULL, arr,
 | 
						|
                                      image, selImage, data);
 | 
						|
    if (HasFlag(wxTR_HIDE_ROOT))
 | 
						|
    {
 | 
						|
        // if root is hidden, make sure we can navigate
 | 
						|
        // into children
 | 
						|
        m_anchor->SetHasPlus();
 | 
						|
        Expand(m_anchor);
 | 
						|
    }
 | 
						|
    if ( data != NULL )
 | 
						|
    {
 | 
						|
        data->SetId((long)m_anchor);
 | 
						|
    }
 | 
						|
 | 
						|
    if (!HasFlag(wxTR_MULTIPLE))
 | 
						|
    {
 | 
						|
        m_current = m_key_current = m_anchor;
 | 
						|
        m_current->SetHilight( TRUE );
 | 
						|
    }
 | 
						|
 | 
						|
    return m_anchor;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::PrependItem(const wxTreeItemId& parent,
 | 
						|
                                     const wxString& text,
 | 
						|
                                     int image, int selImage,
 | 
						|
                                     wxTreeItemData *data)
 | 
						|
{
 | 
						|
    return DoInsertItem(parent, 0u, text, image, selImage, data);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::InsertItem(const wxTreeItemId& parentId,
 | 
						|
                                    const wxTreeItemId& idPrevious,
 | 
						|
                                    const wxString& text,
 | 
						|
                                    int image, int selImage,
 | 
						|
                                    wxTreeItemData *data)
 | 
						|
{
 | 
						|
    wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
 | 
						|
    if ( !parent )
 | 
						|
    {
 | 
						|
        // should we give a warning here?
 | 
						|
        return AddRoot(text, image, selImage, data);
 | 
						|
    }
 | 
						|
 | 
						|
    int index = parent->GetChildren().Index((wxTreeListItem*) idPrevious.m_pItem);
 | 
						|
    wxASSERT_MSG( index != wxNOT_FOUND,
 | 
						|
                  wxT("previous item in wxTreeListMainWindow::InsertItem() is not a sibling") );
 | 
						|
 | 
						|
    return DoInsertItem(parentId, (size_t)++index, text, image, selImage, data);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::InsertItem(const wxTreeItemId& parentId,
 | 
						|
                                    size_t before,
 | 
						|
                                    const wxString& text,
 | 
						|
                                    int image, int selImage,
 | 
						|
                                    wxTreeItemData *data)
 | 
						|
{
 | 
						|
    wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
 | 
						|
    if ( !parent )
 | 
						|
    {
 | 
						|
        // should we give a warning here?
 | 
						|
        return AddRoot(text, image, selImage, data);
 | 
						|
    }
 | 
						|
 | 
						|
    return DoInsertItem(parentId, before, text, image, selImage, data);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxTreeItemId wxTreeListMainWindow::AppendItem(const wxTreeItemId& parentId,
 | 
						|
                                    const wxString& text,
 | 
						|
                                    int image, int selImage,
 | 
						|
                                    wxTreeItemData *data)
 | 
						|
{
 | 
						|
    wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
 | 
						|
    if ( !parent )
 | 
						|
    {
 | 
						|
        // should we give a warning here?
 | 
						|
        return AddRoot(text, image, selImage, data);
 | 
						|
    }
 | 
						|
 | 
						|
    return DoInsertItem( parent, parent->GetChildren().Count(), text,
 | 
						|
                         image, selImage, data);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::SendDeleteEvent(wxTreeListItem *item)
 | 
						|
{
 | 
						|
    wxTreeEvent event( wxEVT_COMMAND_TREE_DELETE_ITEM, m_owner->GetId() );
 | 
						|
    event.SetItem((long) item);
 | 
						|
    event.SetEventObject( /*this*/m_owner );
 | 
						|
    m_owner->ProcessEvent( event );
 | 
						|
}
 | 
						|
 | 
						|
// inline
 | 
						|
// void wxTreeListMainWindow::DeleteChildren(const wxTreeItemId& itemId)
 | 
						|
// {
 | 
						|
//     m_dirty = TRUE;     // do this first so stuff below doesn't cause flicker
 | 
						|
 | 
						|
//     wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
//     item->DeleteChildren(this);
 | 
						|
// }
 | 
						|
 | 
						|
// inline
 | 
						|
// void wxTreeListMainWindow::Delete(const wxTreeItemId& itemId)
 | 
						|
// {
 | 
						|
//     m_dirty = TRUE;     // do this first so stuff below doesn't cause flicker
 | 
						|
 | 
						|
//     wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
//     // don't stay with invalid m_key_current or we will crash in
 | 
						|
//     // the next call to OnChar()
 | 
						|
//     bool changeKeyCurrent = FALSE;
 | 
						|
//     wxTreeListItem *itemKey = m_key_current;
 | 
						|
//     while ( itemKey )
 | 
						|
//     {
 | 
						|
//         if ( itemKey == item )
 | 
						|
//         {
 | 
						|
//             // m_key_current is a descendant of the item being deleted
 | 
						|
//             changeKeyCurrent = TRUE;
 | 
						|
//             break;
 | 
						|
//         }
 | 
						|
//         itemKey = itemKey->GetParent();
 | 
						|
//     }
 | 
						|
 | 
						|
//     wxTreeListItem *parent = item->GetParent();
 | 
						|
//     if ( parent )
 | 
						|
//     {
 | 
						|
//         parent->GetChildren().Remove( item );  // remove by value
 | 
						|
//     }
 | 
						|
 | 
						|
//     if ( changeKeyCurrent )
 | 
						|
//     {
 | 
						|
//         // may be NULL or not
 | 
						|
//         m_key_current = parent;
 | 
						|
//     }
 | 
						|
 | 
						|
//     item->DeleteChildren(this);
 | 
						|
//     SendDeleteEvent(item);
 | 
						|
//     delete item;
 | 
						|
// }
 | 
						|
 | 
						|
// inline
 | 
						|
// void wxTreeListMainWindow::DeleteAllItems()
 | 
						|
// {
 | 
						|
//     if ( m_anchor )
 | 
						|
//     {
 | 
						|
//         m_dirty = TRUE;
 | 
						|
 | 
						|
//         m_anchor->DeleteChildren(this);
 | 
						|
//         delete m_anchor;
 | 
						|
 | 
						|
//         m_anchor = NULL;
 | 
						|
//     }
 | 
						|
// }
 | 
						|
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::DeleteChildren(const wxTreeItemId& itemId)
 | 
						|
{
 | 
						|
    m_dirty = TRUE;     // do this first so stuff below doesn't cause flicker
 | 
						|
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    // mst:16.10.03
 | 
						|
    // moved from Delete()
 | 
						|
    // don't stay with invalid m_key_current or we will crash in
 | 
						|
    // the next call to OnChar()
 | 
						|
    wxTreeListItem *itemKey = m_key_current;
 | 
						|
    while ( itemKey )
 | 
						|
    {
 | 
						|
        if ( itemKey == item )
 | 
						|
        {
 | 
						|
            // m_key_current is a descendant of the item which childrens being deleted
 | 
						|
            m_key_current = item;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
        itemKey = itemKey->GetParent();
 | 
						|
    }
 | 
						|
 | 
						|
    item->DeleteChildren(this);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::Delete(const wxTreeItemId& itemId)
 | 
						|
{
 | 
						|
    m_dirty = TRUE;     // do this first so stuff below doesn't cause flicker
 | 
						|
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    // mst:16.10.03
 | 
						|
    item->DeleteChildren(this);
 | 
						|
 | 
						|
    wxTreeListItem *parent = item->GetParent();
 | 
						|
 | 
						|
    if ( parent )
 | 
						|
       parent->GetChildren().Remove( item );  // remove by value
 | 
						|
 | 
						|
    if (m_key_current == item)
 | 
						|
      m_key_current = parent;
 | 
						|
 | 
						|
    SendDeleteEvent(item);
 | 
						|
 | 
						|
    delete item;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::DeleteAllItems()
 | 
						|
{
 | 
						|
    if ( m_anchor )
 | 
						|
    {
 | 
						|
        m_dirty = TRUE;
 | 
						|
 | 
						|
        m_key_current = NULL;  // mst:16.10.03
 | 
						|
 | 
						|
        m_anchor->DeleteChildren(this);
 | 
						|
        delete m_anchor;
 | 
						|
 | 
						|
        m_anchor = NULL;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void wxTreeListMainWindow::Expand(const wxTreeItemId& itemId)
 | 
						|
{
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    wxCHECK_RET( item, _T("invalid item in wxTreeListMainWindow::Expand") );
 | 
						|
 | 
						|
    if ( !item->HasPlus() )
 | 
						|
        return;
 | 
						|
 | 
						|
    if ( item->IsExpanded() )
 | 
						|
        return;
 | 
						|
 | 
						|
    wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_EXPANDING, m_owner->GetId() );
 | 
						|
    event.SetItem( (long) item );
 | 
						|
    event.SetEventObject( /*this*/m_owner );
 | 
						|
 | 
						|
    if ( m_owner->ProcessEvent( event ) && !event.IsAllowed() )
 | 
						|
    {
 | 
						|
        // cancelled by program
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    item->Expand();
 | 
						|
    CalculatePositions();
 | 
						|
 | 
						|
    RefreshSubtree(item);
 | 
						|
 | 
						|
    event.SetEventType(wxEVT_COMMAND_TREE_ITEM_EXPANDED);
 | 
						|
    ProcessEvent( event );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::ExpandAll(const wxTreeItemId& item)
 | 
						|
{
 | 
						|
    Expand(item);
 | 
						|
    if ( IsExpanded(item) )
 | 
						|
    {
 | 
						|
        long cookie;
 | 
						|
        wxTreeItemId child = GetFirstChild(item, cookie);
 | 
						|
        while ( child.IsOk() )
 | 
						|
        {
 | 
						|
            ExpandAll(child);
 | 
						|
 | 
						|
            child = GetNextChild(item, cookie);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::Collapse(const wxTreeItemId& itemId)
 | 
						|
{
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    if ( !item->IsExpanded() )
 | 
						|
        return;
 | 
						|
 | 
						|
    wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_COLLAPSING, m_owner->GetId() );
 | 
						|
    event.SetItem( (long) item );
 | 
						|
    event.SetEventObject( /*this*/m_owner );
 | 
						|
    if ( m_owner->ProcessEvent( event ) && !event.IsAllowed() )
 | 
						|
    {
 | 
						|
        // cancelled by program
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    item->Collapse();
 | 
						|
 | 
						|
#if 0  // TODO why should items be collapsed recursively?
 | 
						|
    wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
    size_t count = children.Count();
 | 
						|
    for ( size_t n = 0; n < count; n++ )
 | 
						|
    {
 | 
						|
        Collapse(children[n]);
 | 
						|
    }
 | 
						|
#endif
 | 
						|
 | 
						|
    CalculatePositions();
 | 
						|
 | 
						|
    RefreshSubtree(item);
 | 
						|
 | 
						|
    event.SetEventType(wxEVT_COMMAND_TREE_ITEM_COLLAPSED);
 | 
						|
    ProcessEvent( event );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::CollapseAndReset(const wxTreeItemId& item)
 | 
						|
{
 | 
						|
    Collapse(item);
 | 
						|
    DeleteChildren(item);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::Toggle(const wxTreeItemId& itemId)
 | 
						|
{
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    if (item->IsExpanded())
 | 
						|
        Collapse(itemId);
 | 
						|
    else
 | 
						|
        Expand(itemId);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::Unselect()
 | 
						|
{
 | 
						|
    if (m_current)
 | 
						|
    {
 | 
						|
        m_current->SetHilight( FALSE );
 | 
						|
        RefreshLine( m_current );
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::UnselectAllChildren(wxTreeListItem *item)
 | 
						|
{
 | 
						|
    if (item->IsSelected())
 | 
						|
    {
 | 
						|
        item->SetHilight(FALSE);
 | 
						|
        RefreshLine(item);
 | 
						|
    }
 | 
						|
 | 
						|
    if (item->HasChildren())
 | 
						|
    {
 | 
						|
        wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
        size_t count = children.Count();
 | 
						|
        for ( size_t n = 0; n < count; ++n )
 | 
						|
        {
 | 
						|
            UnselectAllChildren(children[n]);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::UnselectAll()
 | 
						|
{
 | 
						|
    UnselectAllChildren((wxTreeListItem*) GetRootItem().m_pItem);
 | 
						|
}
 | 
						|
 | 
						|
// Recursive function !
 | 
						|
// To stop we must have crt_item<last_item
 | 
						|
// Algorithm :
 | 
						|
// Tag all next children, when no more children,
 | 
						|
// Move to parent (not to tag)
 | 
						|
// Keep going... if we found last_item, we stop.
 | 
						|
bool wxTreeListMainWindow::TagNextChildren(wxTreeListItem *crt_item, wxTreeListItem *last_item, bool select)
 | 
						|
{
 | 
						|
    wxTreeListItem *parent = crt_item->GetParent();
 | 
						|
 | 
						|
    if (parent == NULL) // This is root item
 | 
						|
        return TagAllChildrenUntilLast(crt_item, last_item, select);
 | 
						|
 | 
						|
    wxArrayTreeListItems& children = parent->GetChildren();
 | 
						|
    int index = children.Index(crt_item);
 | 
						|
    wxASSERT( index != wxNOT_FOUND ); // I'm not a child of my parent?
 | 
						|
 | 
						|
    size_t count = children.Count();
 | 
						|
    for (size_t n=(size_t)(index+1); n<count; ++n)
 | 
						|
    {
 | 
						|
        if (TagAllChildrenUntilLast(children[n], last_item, select)) return TRUE;
 | 
						|
    }
 | 
						|
 | 
						|
    return TagNextChildren(parent, last_item, select);
 | 
						|
}
 | 
						|
 | 
						|
bool wxTreeListMainWindow::TagAllChildrenUntilLast(wxTreeListItem *crt_item, wxTreeListItem *last_item, bool select)
 | 
						|
{
 | 
						|
    crt_item->SetHilight(select);
 | 
						|
    RefreshLine(crt_item);
 | 
						|
 | 
						|
    if (crt_item==last_item)
 | 
						|
        return TRUE;
 | 
						|
 | 
						|
    if (crt_item->HasChildren())
 | 
						|
    {
 | 
						|
        wxArrayTreeListItems& children = crt_item->GetChildren();
 | 
						|
        size_t count = children.Count();
 | 
						|
        for ( size_t n = 0; n < count; ++n )
 | 
						|
        {
 | 
						|
            if (TagAllChildrenUntilLast(children[n], last_item, select))
 | 
						|
                return TRUE;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
  return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::SelectItemRange(wxTreeListItem *item1, wxTreeListItem *item2)
 | 
						|
{
 | 
						|
    // item2 is not necessary after item1
 | 
						|
    wxTreeListItem *first=NULL, *last=NULL;
 | 
						|
 | 
						|
    // choice first' and 'last' between item1 and item2
 | 
						|
    if (item1->GetY()<item2->GetY())
 | 
						|
    {
 | 
						|
        first=item1;
 | 
						|
        last=item2;
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        first=item2;
 | 
						|
        last=item1;
 | 
						|
    }
 | 
						|
 | 
						|
    bool select = m_current->IsSelected();
 | 
						|
 | 
						|
    if ( TagAllChildrenUntilLast(first,last,select) )
 | 
						|
        return;
 | 
						|
 | 
						|
    TagNextChildren(first,last,select);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::SelectItem(const wxTreeItemId& itemId,
 | 
						|
                            bool unselect_others,
 | 
						|
                            bool extended_select)
 | 
						|
{
 | 
						|
    wxCHECK_RET( itemId.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    bool is_single=!(GetWindowStyleFlag() & wxTR_MULTIPLE);
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    //wxCHECK_RET( ( (!unselect_others) && is_single),
 | 
						|
    //           wxT("this is a single selection tree") );
 | 
						|
 | 
						|
    // to keep going anyhow !!!
 | 
						|
    if (is_single)
 | 
						|
    {
 | 
						|
        if (item->IsSelected())
 | 
						|
            return; // nothing to do
 | 
						|
        unselect_others = TRUE;
 | 
						|
        extended_select = FALSE;
 | 
						|
    }
 | 
						|
    else if ( unselect_others && item->IsSelected() )
 | 
						|
    {
 | 
						|
        // selection change if there is more than one item currently selected
 | 
						|
        wxArrayTreeItemIds selected_items;
 | 
						|
        if ( GetSelections(selected_items) == 1 )
 | 
						|
            return;
 | 
						|
    }
 | 
						|
 | 
						|
    wxTreeEvent event( wxEVT_COMMAND_TREE_SEL_CHANGING, m_owner->GetId() );
 | 
						|
    event.SetItem( (long) item );
 | 
						|
    event.SetOldItem( (long) m_current );
 | 
						|
    event.SetEventObject( /*this*/m_owner );
 | 
						|
    // TODO : Here we don't send any selection mode yet !
 | 
						|
 | 
						|
    if(m_owner->GetEventHandler()->ProcessEvent( event ) && !event.IsAllowed())
 | 
						|
        return;
 | 
						|
 | 
						|
    wxTreeItemId parent = GetParent( itemId );
 | 
						|
    while (parent.IsOk())
 | 
						|
    {
 | 
						|
        if (!IsExpanded(parent))
 | 
						|
            Expand( parent );
 | 
						|
 | 
						|
        parent = GetParent( parent );
 | 
						|
    }
 | 
						|
 | 
						|
    EnsureVisible( itemId );
 | 
						|
 | 
						|
    // ctrl press
 | 
						|
    if (unselect_others)
 | 
						|
    {
 | 
						|
        if (is_single) Unselect(); // to speed up thing
 | 
						|
        else UnselectAll();
 | 
						|
    }
 | 
						|
 | 
						|
    // shift press
 | 
						|
    if (extended_select)
 | 
						|
    {
 | 
						|
        if ( !m_current )
 | 
						|
        {
 | 
						|
            m_current = m_key_current = (wxTreeListItem*) GetRootItem().m_pItem;
 | 
						|
        }
 | 
						|
 | 
						|
        // don't change the mark (m_current)
 | 
						|
        SelectItemRange(m_current, item);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        bool select=TRUE; // the default
 | 
						|
 | 
						|
        // Check if we need to toggle hilight (ctrl mode)
 | 
						|
        if (!unselect_others)
 | 
						|
            select=!item->IsSelected();
 | 
						|
 | 
						|
        m_current = m_key_current = item;
 | 
						|
        m_current->SetHilight(select);
 | 
						|
        RefreshLine( m_current );
 | 
						|
    }
 | 
						|
 | 
						|
    event.SetEventType(wxEVT_COMMAND_TREE_SEL_CHANGED);
 | 
						|
    GetEventHandler()->ProcessEvent( event );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::FillArray(wxTreeListItem *item,
 | 
						|
                           wxArrayTreeItemIds &array) const
 | 
						|
{
 | 
						|
    if ( item->IsSelected() )
 | 
						|
        array.Add(wxTreeItemId(item));
 | 
						|
 | 
						|
    if ( item->HasChildren() )
 | 
						|
    {
 | 
						|
        wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
        size_t count = children.GetCount();
 | 
						|
        for ( size_t n = 0; n < count; ++n )
 | 
						|
            FillArray(children[n], array);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
size_t wxTreeListMainWindow::GetSelections(wxArrayTreeItemIds &array) const
 | 
						|
{
 | 
						|
    array.Empty();
 | 
						|
    wxTreeItemId idRoot = GetRootItem();
 | 
						|
    if ( idRoot.IsOk() )
 | 
						|
    {
 | 
						|
        FillArray((wxTreeListItem*) idRoot.m_pItem, array);
 | 
						|
    }
 | 
						|
    //else: the tree is empty, so no selections
 | 
						|
 | 
						|
    return array.Count();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::EnsureVisible(const wxTreeItemId& item)
 | 
						|
{
 | 
						|
    if (!item.IsOk()) return;
 | 
						|
 | 
						|
    wxTreeListItem *gitem = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    // first expand all parent branches
 | 
						|
    wxTreeListItem *parent = gitem->GetParent();
 | 
						|
    while ( parent )
 | 
						|
    {
 | 
						|
        Expand(parent);
 | 
						|
        parent = parent->GetParent();
 | 
						|
    }
 | 
						|
 | 
						|
    //if (parent) CalculatePositions();
 | 
						|
 | 
						|
    ScrollTo(item);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::ScrollTo(const wxTreeItemId &item)
 | 
						|
{
 | 
						|
    if (!item.IsOk()) return;
 | 
						|
 | 
						|
    // We have to call this here because the label in
 | 
						|
    // question might just have been added and no screen
 | 
						|
    // update taken place.
 | 
						|
    if (m_dirty) wxYieldIfNeeded();
 | 
						|
 | 
						|
    wxTreeListItem *gitem = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    // now scroll to the item
 | 
						|
    int item_y = gitem->GetY();
 | 
						|
 | 
						|
    int start_x = 0;
 | 
						|
    int start_y = 0;
 | 
						|
    GetViewStart( &start_x, &start_y );
 | 
						|
    start_y *= PIXELS_PER_UNIT;
 | 
						|
 | 
						|
    int client_h = 0;
 | 
						|
    int client_w = 0;
 | 
						|
    GetClientSize( &client_w, &client_h );
 | 
						|
 | 
						|
    if (item_y < start_y+3)
 | 
						|
    {
 | 
						|
        // going down
 | 
						|
        int x = 0;
 | 
						|
        int y = 0;
 | 
						|
        m_anchor->GetSize( x, y, this );
 | 
						|
        x = m_owner->GetHeaderWindow()->GetWidth(); //m_total_col_width; // ALB
 | 
						|
        y += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
 | 
						|
        //x += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
 | 
						|
        int x_pos = GetScrollPos( wxHORIZONTAL );
 | 
						|
        // Item should appear at top
 | 
						|
        SetScrollbars( PIXELS_PER_UNIT, PIXELS_PER_UNIT, x/PIXELS_PER_UNIT, y/PIXELS_PER_UNIT, x_pos, item_y/PIXELS_PER_UNIT );
 | 
						|
    }
 | 
						|
    else if (item_y+GetLineHeight(gitem) > start_y+client_h)
 | 
						|
    {
 | 
						|
        // going up
 | 
						|
        int x = 0;
 | 
						|
        int y = 0;
 | 
						|
        m_anchor->GetSize( x, y, this );
 | 
						|
        y += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
 | 
						|
        //x += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
 | 
						|
        x = m_owner->GetHeaderWindow()->GetWidth(); //m_total_col_width; // ALB
 | 
						|
        item_y += PIXELS_PER_UNIT+2;
 | 
						|
        int x_pos = GetScrollPos( wxHORIZONTAL );
 | 
						|
        // Item should appear at bottom
 | 
						|
        SetScrollbars( PIXELS_PER_UNIT, PIXELS_PER_UNIT, x/PIXELS_PER_UNIT, y/PIXELS_PER_UNIT, x_pos, (item_y+GetLineHeight(gitem)-client_h)/PIXELS_PER_UNIT );
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
// FIXME: tree sorting functions are not reentrant and not MT-safe!
 | 
						|
static wxTreeListMainWindow *s_treeBeingSorted = NULL;
 | 
						|
 | 
						|
static int LINKAGEMODE tree_ctrl_compare_func(wxTreeListItem **item1,
 | 
						|
                                  wxTreeListItem **item2)
 | 
						|
{
 | 
						|
    wxCHECK_MSG( s_treeBeingSorted, 0, wxT("bug in wxTreeListMainWindow::SortChildren()") );
 | 
						|
 | 
						|
    return s_treeBeingSorted->OnCompareItems(*item1, *item2);
 | 
						|
}
 | 
						|
 | 
						|
int wxTreeListMainWindow::OnCompareItems(const wxTreeItemId& item1,
 | 
						|
                               const wxTreeItemId& item2)
 | 
						|
{
 | 
						|
    // ALB: delegate to m_owner, to let the user overrride the comparison
 | 
						|
    //return wxStrcmp(GetItemText(item1), GetItemText(item2));
 | 
						|
    return m_owner->OnCompareItems(item1, item2);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::SortChildren(const wxTreeItemId& itemId)
 | 
						|
{
 | 
						|
    wxCHECK_RET( itemId.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
 | 
						|
 | 
						|
    wxCHECK_RET( !s_treeBeingSorted,
 | 
						|
                 wxT("wxTreeListMainWindow::SortChildren is not reentrant") );
 | 
						|
 | 
						|
    wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
    if ( children.Count() > 1 )
 | 
						|
    {
 | 
						|
        m_dirty = TRUE;
 | 
						|
 | 
						|
        s_treeBeingSorted = this;
 | 
						|
        children.Sort(tree_ctrl_compare_func);
 | 
						|
        s_treeBeingSorted = NULL;
 | 
						|
    }
 | 
						|
    //else: don't make the tree dirty as nothing changed
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxImageList *wxTreeListMainWindow::GetImageList() const
 | 
						|
{
 | 
						|
    return m_imageListNormal;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxImageList *wxTreeListMainWindow::GetButtonsImageList() const
 | 
						|
{
 | 
						|
    return m_imageListButtons;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxImageList *wxTreeListMainWindow::GetStateImageList() const
 | 
						|
{
 | 
						|
    return m_imageListState;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::CalculateLineHeight()
 | 
						|
{
 | 
						|
    wxClientDC dc(this);
 | 
						|
    m_lineHeight = (int)(dc.GetCharHeight() + m_linespacing*2);
 | 
						|
 | 
						|
    if ( m_imageListNormal )
 | 
						|
    {
 | 
						|
        // Calculate a m_lineHeight value from the normal Image sizes.
 | 
						|
        // May be toggle off. Then wxTreeListMainWindow will spread when
 | 
						|
        // necessary (which might look ugly).
 | 
						|
        int n = m_imageListNormal->GetImageCount();
 | 
						|
        for (int i = 0; i < n ; i++)
 | 
						|
        {
 | 
						|
            int width = 0, height = 0;
 | 
						|
            m_imageListNormal->GetSize(i, width, height);
 | 
						|
            if (height > m_lineHeight) m_lineHeight = height;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (m_imageListButtons)
 | 
						|
    {
 | 
						|
        // Calculate a m_lineHeight value from the Button image sizes.
 | 
						|
        // May be toggle off. Then wxTreeListMainWindow will spread when
 | 
						|
        // necessary (which might look ugly).
 | 
						|
        int n = m_imageListButtons->GetImageCount();
 | 
						|
        for (int i = 0; i < n ; i++)
 | 
						|
        {
 | 
						|
            int width = 0, height = 0;
 | 
						|
            m_imageListButtons->GetSize(i, width, height);
 | 
						|
            if (height > m_lineHeight) m_lineHeight = height;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (m_lineHeight < 30)
 | 
						|
        m_lineHeight += 2;                 // at least 2 pixels
 | 
						|
    else
 | 
						|
        m_lineHeight += m_lineHeight/10;   // otherwise 10% extra spacing
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetImageList(wxImageList *imageList)
 | 
						|
{
 | 
						|
    if (m_ownsImageListNormal) delete m_imageListNormal;
 | 
						|
    m_imageListNormal = imageList;
 | 
						|
    m_ownsImageListNormal = FALSE;
 | 
						|
    m_dirty = TRUE;
 | 
						|
    CalculateLineHeight();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetStateImageList(wxImageList *imageList)
 | 
						|
{
 | 
						|
    if (m_ownsImageListState) delete m_imageListState;
 | 
						|
    m_imageListState = imageList;
 | 
						|
    m_ownsImageListState = FALSE;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetButtonsImageList(wxImageList *imageList)
 | 
						|
{
 | 
						|
    if (m_ownsImageListButtons) delete m_imageListButtons;
 | 
						|
    m_imageListButtons = imageList;
 | 
						|
    m_ownsImageListButtons = FALSE;
 | 
						|
    m_dirty = TRUE;
 | 
						|
    CalculateLineHeight();
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::AssignImageList(wxImageList *imageList)
 | 
						|
{
 | 
						|
    SetImageList(imageList);
 | 
						|
    m_ownsImageListNormal = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::AssignStateImageList(wxImageList *imageList)
 | 
						|
{
 | 
						|
    SetStateImageList(imageList);
 | 
						|
    m_ownsImageListState = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::AssignButtonsImageList(wxImageList *imageList)
 | 
						|
{
 | 
						|
    SetButtonsImageList(imageList);
 | 
						|
    m_ownsImageListButtons = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// helpers
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
void wxTreeListMainWindow::AdjustMyScrollbars()
 | 
						|
{
 | 
						|
    if (m_anchor)
 | 
						|
    {
 | 
						|
        int x = 0, y = 0;
 | 
						|
        m_anchor->GetSize( x, y, this );
 | 
						|
        y += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
 | 
						|
        //x += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
 | 
						|
        int x_pos = GetScrollPos( wxHORIZONTAL );
 | 
						|
        int y_pos = GetScrollPos( wxVERTICAL );
 | 
						|
        x = m_owner->GetHeaderWindow()->GetWidth() + 2;
 | 
						|
        if(x < GetClientSize().GetWidth()) x_pos = 0;
 | 
						|
        //m_total_col_width + 2; // ALB
 | 
						|
        SetScrollbars( PIXELS_PER_UNIT, PIXELS_PER_UNIT, x/PIXELS_PER_UNIT,
 | 
						|
                       y/PIXELS_PER_UNIT, x_pos, y_pos );
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        SetScrollbars( 0, 0, 0, 0 );
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
int wxTreeListMainWindow::GetLineHeight(wxTreeListItem *item) const
 | 
						|
{
 | 
						|
    if (GetWindowStyleFlag() & wxTR_HAS_VARIABLE_ROW_HEIGHT)
 | 
						|
        return item->GetHeight();
 | 
						|
    else
 | 
						|
        return m_lineHeight;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::PaintItem(wxTreeListItem *item, wxDC& dc)
 | 
						|
{
 | 
						|
    // TODO implement "state" icon on items
 | 
						|
 | 
						|
    wxTreeItemAttr *attr = item->GetAttributes();
 | 
						|
    if ( attr && attr->HasFont() )
 | 
						|
        dc.SetFont(attr->GetFont());
 | 
						|
    else if (item->IsBold())
 | 
						|
        dc.SetFont(m_boldFont);
 | 
						|
 | 
						|
    long text_w = 0, text_h = 0;
 | 
						|
 | 
						|
    dc.GetTextExtent( item->GetText(GetMainColumn()), &text_w, &text_h );
 | 
						|
 | 
						|
    int total_h = GetLineHeight(item);
 | 
						|
 | 
						|
    if ( item->IsSelected() )
 | 
						|
    {
 | 
						|
        dc.SetBrush(*(m_hasFocus ? m_hilightBrush : m_hilightUnfocusedBrush));
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        wxColour colBg;
 | 
						|
        if ( attr && attr->HasBackgroundColour() )
 | 
						|
            colBg = attr->GetBackgroundColour();
 | 
						|
        else
 | 
						|
            colBg = m_backgroundColour;
 | 
						|
        dc.SetBrush(wxBrush(colBg, wxSOLID));
 | 
						|
    }
 | 
						|
 | 
						|
    int offset = HasFlag(wxTR_ROW_LINES) ? 1 : 0;
 | 
						|
 | 
						|
    dc.DrawRectangle(0, item->GetY()+offset,
 | 
						|
                     m_owner->GetHeaderWindow()->GetWidth(),
 | 
						|
                     total_h-offset);
 | 
						|
 | 
						|
    dc.SetBackgroundMode(wxTRANSPARENT);
 | 
						|
    int extraH = (total_h > text_h) ? (total_h - text_h)/2 : 0;
 | 
						|
    int extra_offset = 0;
 | 
						|
    for(size_t i = 0; i < GetColumnCount(); ++i) {
 | 
						|
        int coord_x = extra_offset, image_x = coord_x;
 | 
						|
        int clip_width = m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
        int image_h = 0, image_w = 0; //2;
 | 
						|
        int image = NO_IMAGE;
 | 
						|
 | 
						|
        if(i == GetMainColumn()) {
 | 
						|
            image = item->GetCurrentImage();
 | 
						|
            coord_x = item->GetX();
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            image = item->GetImage(i);
 | 
						|
        }
 | 
						|
 | 
						|
        if(image != NO_IMAGE) {
 | 
						|
            if(m_imageListNormal) {
 | 
						|
                m_imageListNormal->GetSize( image, image_w, image_h );
 | 
						|
                image_w += 4;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                image = NO_IMAGE;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // honor text alignment
 | 
						|
        wxString text = item->GetText(i);
 | 
						|
 | 
						|
        switch(m_owner->GetHeaderWindow()->GetColumn(i).GetAlignment()) {
 | 
						|
        case wxTL_ALIGN_LEFT:
 | 
						|
            coord_x += image_w + 2;
 | 
						|
            image_x = coord_x - image_w;
 | 
						|
            break;
 | 
						|
        case wxTL_ALIGN_RIGHT:
 | 
						|
            dc.GetTextExtent(text, &text_w, NULL);
 | 
						|
            coord_x += clip_width - text_w - image_w - 2;
 | 
						|
            image_x = coord_x - image_w;
 | 
						|
            break;
 | 
						|
        case wxTL_ALIGN_CENTER:
 | 
						|
            dc.GetTextExtent(text, &text_w, NULL);
 | 
						|
            //coord_x += (clip_width - text_w)/2 + image_w;
 | 
						|
            image_x += (clip_width - text_w - image_w)/2 + 2;
 | 
						|
            coord_x = image_x + image_w;
 | 
						|
        }
 | 
						|
 | 
						|
        wxDCClipper clipper(dc, /*coord_x,*/ extra_offset,
 | 
						|
                            item->GetY() + extraH, clip_width,
 | 
						|
                            total_h);
 | 
						|
 | 
						|
        if(image != NO_IMAGE) {
 | 
						|
            m_imageListNormal->Draw( image, dc, image_x,
 | 
						|
                                     item->GetY() +((total_h > image_h)?
 | 
						|
                                                    ((total_h-image_h)/2):0),
 | 
						|
                                     wxIMAGELIST_DRAW_TRANSPARENT );
 | 
						|
        }
 | 
						|
 | 
						|
        dc.DrawText( text,
 | 
						|
                     (wxCoord)(coord_x /*image_w + item->GetX()*/),
 | 
						|
                     (wxCoord)(item->GetY() + extraH));
 | 
						|
        extra_offset += m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
    }
 | 
						|
 | 
						|
    // restore normal font
 | 
						|
    dc.SetFont( m_normalFont );
 | 
						|
}
 | 
						|
 | 
						|
// Now y stands for the top of the item, whereas it used to stand for middle !
 | 
						|
void wxTreeListMainWindow::PaintLevel( wxTreeListItem *item, wxDC &dc,
 | 
						|
                                    int level, int &y, int x_offset )
 | 
						|
{
 | 
						|
    int x = level*m_indent + x_offset;
 | 
						|
    if (!HasFlag(wxTR_HIDE_ROOT))
 | 
						|
    {
 | 
						|
        x += m_indent;
 | 
						|
    }
 | 
						|
    else if (level == 0)
 | 
						|
    {
 | 
						|
        // always expand hidden root
 | 
						|
        int origY = y;
 | 
						|
        wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
        int count = children.Count();
 | 
						|
        if (count > 0)
 | 
						|
        {
 | 
						|
            int n = 0, oldY;
 | 
						|
            do {
 | 
						|
                oldY = y;
 | 
						|
                PaintLevel(children[n], dc, 1, y, x_offset);
 | 
						|
            } while (++n < count);
 | 
						|
 | 
						|
            if (!HasFlag(wxTR_NO_LINES) && HasFlag(wxTR_LINES_AT_ROOT) &&
 | 
						|
                count > 0)
 | 
						|
            {
 | 
						|
                // draw line down to last child
 | 
						|
                origY += GetLineHeight(children[0])>>1;
 | 
						|
                oldY += GetLineHeight(children[n-1])>>1;
 | 
						|
                dc.DrawLine(3, origY, 3, oldY);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    item->SetX(x+m_spacing);
 | 
						|
    item->SetY(y);
 | 
						|
 | 
						|
    int h = GetLineHeight(item);
 | 
						|
    int y_top = y;
 | 
						|
    int y_mid = y_top + (h>>1);
 | 
						|
    y += h;
 | 
						|
 | 
						|
    int exposed_x = dc.LogicalToDeviceX(0);
 | 
						|
    int exposed_y = dc.LogicalToDeviceY(y_top);
 | 
						|
 | 
						|
    if (IsExposed(exposed_x, exposed_y, 10000, h))  // 10000 = very much
 | 
						|
    {
 | 
						|
        wxPen *pen =
 | 
						|
#ifndef __WXMAC__
 | 
						|
            // don't draw rect outline if we already have the
 | 
						|
            // background color under Mac
 | 
						|
            (item->IsSelected() && m_hasFocus) ? wxBLACK_PEN :
 | 
						|
#endif // !__WXMAC__
 | 
						|
            wxTRANSPARENT_PEN;
 | 
						|
 | 
						|
        wxColour colText;
 | 
						|
        if ( item->IsSelected() )
 | 
						|
        {
 | 
						|
            colText = wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHTTEXT);
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            wxTreeItemAttr *attr = item->GetAttributes();
 | 
						|
            if (attr && attr->HasTextColour())
 | 
						|
                colText = attr->GetTextColour();
 | 
						|
            else
 | 
						|
                //colText = wxSystemSettings::GetSystemColour(wxSYS_COLOUR_WINDOWTEXT);
 | 
						|
                colText = GetForegroundColour();
 | 
						|
        }
 | 
						|
 | 
						|
        // prepare to draw
 | 
						|
        dc.SetTextForeground(colText);
 | 
						|
        dc.SetPen(*pen);
 | 
						|
 | 
						|
        // draw
 | 
						|
        PaintItem(item, dc);
 | 
						|
 | 
						|
        if (HasFlag(wxTR_ROW_LINES))
 | 
						|
        {
 | 
						|
            int total_width = m_owner->GetHeaderWindow()->GetWidth();
 | 
						|
            // if the background colour is white, choose a
 | 
						|
            // contrasting color for the lines
 | 
						|
            dc.SetPen(*((GetBackgroundColour() == *wxWHITE)
 | 
						|
                         ? wxMEDIUM_GREY_PEN : wxWHITE_PEN));
 | 
						|
            dc.DrawLine(0, y_top, total_width, y_top);
 | 
						|
            dc.DrawLine(0, y, total_width, y);
 | 
						|
        }
 | 
						|
 | 
						|
        // restore DC objects
 | 
						|
        dc.SetBrush(*wxWHITE_BRUSH);
 | 
						|
        dc.SetPen(m_dottedPen);
 | 
						|
        dc.SetTextForeground(*wxBLACK);
 | 
						|
 | 
						|
        size_t clip_width = m_owner->GetHeaderWindow()->GetColumn(
 | 
						|
            m_main_column).GetWidth();
 | 
						|
            //m_columns[m_main_column].GetWidth();
 | 
						|
        if (item->HasPlus() && HasButtons())  // should the item show a button?
 | 
						|
        {
 | 
						|
            // clip to the column width
 | 
						|
            wxDCClipper clipper(dc, x_offset, y_top, clip_width, 10000);
 | 
						|
 | 
						|
            if (!HasFlag(wxTR_NO_LINES))
 | 
						|
            {
 | 
						|
                if (x > (signed)m_indent)
 | 
						|
                    dc.DrawLine(x - m_indent, y_mid, x - 5, y_mid);
 | 
						|
                else if (HasFlag(wxTR_LINES_AT_ROOT))
 | 
						|
                    dc.DrawLine(3, y_mid, x - 5, y_mid);
 | 
						|
                dc.DrawLine(x + 5, y_mid, x + m_spacing, y_mid);
 | 
						|
            }
 | 
						|
 | 
						|
            if (m_imageListButtons != NULL)
 | 
						|
            {
 | 
						|
                // draw the image button here
 | 
						|
                int image_h = 0, image_w = 0, image = wxTreeItemIcon_Normal;
 | 
						|
                if (item->IsExpanded()) image = wxTreeItemIcon_Expanded;
 | 
						|
                if (item->IsSelected())
 | 
						|
                    image += wxTreeItemIcon_Selected - wxTreeItemIcon_Normal;
 | 
						|
                m_imageListButtons->GetSize(image, image_w, image_h);
 | 
						|
                int xx = x - (image_w>>1);
 | 
						|
                int yy = y_mid - (image_h>>1);
 | 
						|
                dc.SetClippingRegion(xx, yy, image_w, image_h);
 | 
						|
                m_imageListButtons->Draw(image, dc, xx, yy,
 | 
						|
                                         wxIMAGELIST_DRAW_TRANSPARENT);
 | 
						|
                dc.DestroyClippingRegion();
 | 
						|
            }
 | 
						|
            else if (HasFlag(wxTR_TWIST_BUTTONS))
 | 
						|
            {
 | 
						|
                // draw the twisty button here
 | 
						|
                dc.SetPen(*wxBLACK_PEN);
 | 
						|
                dc.SetBrush(*m_hilightBrush);
 | 
						|
 | 
						|
                wxPoint button[3];
 | 
						|
 | 
						|
                if (item->IsExpanded())
 | 
						|
                {
 | 
						|
                    button[0].x = x-5;
 | 
						|
                    button[0].y = y_mid-2;
 | 
						|
                    button[1].x = x+5;
 | 
						|
                    button[1].y = y_mid-2;
 | 
						|
                    button[2].x = x;
 | 
						|
                    button[2].y = y_mid+3;
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    button[0].y = y_mid-5;
 | 
						|
                    button[0].x = x-2;
 | 
						|
                    button[1].y = y_mid+5;
 | 
						|
                    button[1].x = x-2;
 | 
						|
                    button[2].y = y_mid;
 | 
						|
                    button[2].x = x+3;
 | 
						|
                }
 | 
						|
                dc.DrawPolygon(3, button);
 | 
						|
 | 
						|
                dc.SetPen(m_dottedPen);
 | 
						|
            }
 | 
						|
            else // if (HasFlag(wxTR_HAS_BUTTONS))
 | 
						|
            {
 | 
						|
                // draw the plus sign here
 | 
						|
                dc.SetPen(*wxGREY_PEN);
 | 
						|
                dc.SetBrush(*wxWHITE_BRUSH);
 | 
						|
                dc.DrawRectangle(x-5, y_mid-4, 11, 9);
 | 
						|
                dc.SetPen(*wxBLACK_PEN);
 | 
						|
                dc.DrawLine(x-2, y_mid, x+3, y_mid);
 | 
						|
                if (!item->IsExpanded())
 | 
						|
                    dc.DrawLine(x, y_mid-2, x, y_mid+3);
 | 
						|
                dc.SetPen(m_dottedPen);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else if (!HasFlag(wxTR_NO_LINES))  // no button; maybe a line?
 | 
						|
        {
 | 
						|
            // clip to the column width
 | 
						|
            wxDCClipper clipper(dc, x_offset, y_top, clip_width, 10000);
 | 
						|
            // draw the horizontal line here
 | 
						|
            int x_start = x;
 | 
						|
            if (x > (signed)m_indent)
 | 
						|
                x_start -= m_indent;
 | 
						|
            else if (HasFlag(wxTR_LINES_AT_ROOT))
 | 
						|
                x_start = 3;
 | 
						|
            dc.DrawLine(x_start, y_mid, x + m_spacing, y_mid);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (item->IsExpanded())
 | 
						|
    {
 | 
						|
        wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
        int count = children.Count();
 | 
						|
        if (count > 0)
 | 
						|
        {
 | 
						|
            int n = 0, oldY;
 | 
						|
            ++level;
 | 
						|
            do {
 | 
						|
                oldY = y;
 | 
						|
                PaintLevel(children[n], dc, level, y, x_offset);
 | 
						|
            } while (++n < count);
 | 
						|
 | 
						|
            if (!HasFlag(wxTR_NO_LINES) && count > 0)
 | 
						|
            {
 | 
						|
                size_t clip_width = m_owner->GetHeaderWindow()->GetColumn(
 | 
						|
                    m_main_column).GetWidth();
 | 
						|
                    //m_columns[m_main_column].GetWidth();
 | 
						|
                // clip to the column width
 | 
						|
                wxDCClipper clipper(dc, x_offset, y_top, clip_width, 10000);
 | 
						|
                // draw line down to last child
 | 
						|
                oldY += GetLineHeight(children[n-1])>>1;
 | 
						|
                if (HasButtons()) y_mid += 5;
 | 
						|
                dc.DrawLine(x, y_mid, x, oldY);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::DrawDropEffect(wxTreeListItem *item)
 | 
						|
{
 | 
						|
    if ( item )
 | 
						|
    {
 | 
						|
        if ( item->HasPlus() )
 | 
						|
        {
 | 
						|
            // it's a folder, indicate it by a border
 | 
						|
            DrawBorder(item);
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            // draw a line under the drop target because the item will be
 | 
						|
            // dropped there
 | 
						|
            DrawLine(item, TRUE /* below */);
 | 
						|
        }
 | 
						|
 | 
						|
        SetCursor(wxCURSOR_BULLSEYE);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        // can't drop here
 | 
						|
        SetCursor(wxCURSOR_NO_ENTRY);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::DrawBorder(const wxTreeItemId &item)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), _T("invalid item in wxTreeListMainWindow::DrawLine") );
 | 
						|
 | 
						|
    wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC( dc );
 | 
						|
    dc.SetLogicalFunction(wxINVERT);
 | 
						|
    dc.SetBrush(*wxTRANSPARENT_BRUSH);
 | 
						|
 | 
						|
    int w = i->GetWidth() + 2;
 | 
						|
    int h = GetLineHeight(i) + 2;
 | 
						|
 | 
						|
    dc.DrawRectangle( i->GetX() - 1, i->GetY() - 1, w, h);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::DrawLine(const wxTreeItemId &item, bool below)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), _T("invalid item in wxTreeListMainWindow::DrawLine") );
 | 
						|
 | 
						|
    wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC( dc );
 | 
						|
    dc.SetLogicalFunction(wxINVERT);
 | 
						|
 | 
						|
    int x = i->GetX(),
 | 
						|
        y = i->GetY();
 | 
						|
    if ( below )
 | 
						|
    {
 | 
						|
        y += GetLineHeight(i) - 1;
 | 
						|
    }
 | 
						|
 | 
						|
    dc.DrawLine( x, y, x + i->GetWidth(), y);
 | 
						|
}
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// wxWindows callbacks
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
 | 
						|
{
 | 
						|
    wxPaintDC dc(this);
 | 
						|
 | 
						|
    PrepareDC( dc );
 | 
						|
 | 
						|
    if(!GetColumnCount()) return; // ALB
 | 
						|
 | 
						|
    if ( !m_anchor)
 | 
						|
        return;
 | 
						|
 | 
						|
    dc.SetFont( m_normalFont );
 | 
						|
    dc.SetPen( m_dottedPen );
 | 
						|
 | 
						|
    // this is now done dynamically
 | 
						|
    //if(GetImageList() == NULL)
 | 
						|
    // m_lineHeight = (int)(dc.GetCharHeight() + 4);
 | 
						|
 | 
						|
    int y = 0; //HEADER_HEIGHT; //2;
 | 
						|
    int x_offset = 0;
 | 
						|
    for(size_t i = 0; i < GetMainColumn(); ++i) {
 | 
						|
        x_offset += m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
    }
 | 
						|
    PaintLevel( m_anchor, dc, 0, y, x_offset );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnSetFocus( wxFocusEvent &event )
 | 
						|
{
 | 
						|
    m_hasFocus = TRUE;
 | 
						|
 | 
						|
    RefreshSelected();
 | 
						|
 | 
						|
    event.Skip();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnKillFocus( wxFocusEvent &event )
 | 
						|
{
 | 
						|
    m_hasFocus = FALSE;
 | 
						|
 | 
						|
    RefreshSelected();
 | 
						|
 | 
						|
    event.Skip();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnChar( wxKeyEvent &event )
 | 
						|
{
 | 
						|
    wxTreeEvent te( wxEVT_COMMAND_TREE_KEY_DOWN, m_owner->GetId() );
 | 
						|
    te.SetKeyEvent( event );
 | 
						|
    te.SetEventObject( /*this*/m_owner );
 | 
						|
    if ( m_owner->GetEventHandler()->ProcessEvent( te ) )
 | 
						|
    {
 | 
						|
        // intercepted by the user code
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if ( (m_current == 0) || (m_key_current == 0) )
 | 
						|
    {
 | 
						|
        event.Skip();
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // how should the selection work for this event?
 | 
						|
    bool is_multiple, extended_select, unselect_others;
 | 
						|
    EventFlagsToSelType(GetWindowStyleFlag(),
 | 
						|
                        event.ShiftDown(),
 | 
						|
                        event.ControlDown(),
 | 
						|
                        is_multiple, extended_select, unselect_others);
 | 
						|
 | 
						|
    // + : Expand (not on Win32)
 | 
						|
    // - : Collaspe (not on Win32)
 | 
						|
    // * : Expand all/Collapse all
 | 
						|
    // ' ' | return : activate
 | 
						|
    // up    : go up (not last children!)
 | 
						|
    // down  : go down
 | 
						|
    // left  : go to parent (or collapse on Win32)
 | 
						|
    // right : open if parent and go next (or expand on Win32)
 | 
						|
    // home  : go to root
 | 
						|
    // end   : go to last item without opening parents
 | 
						|
    switch (event.GetKeyCode())
 | 
						|
    {
 | 
						|
#ifndef __WXMSW__ // mimic the standard win32 tree ctrl
 | 
						|
        case '+':
 | 
						|
        case WXK_ADD:
 | 
						|
            if (m_current->HasPlus() && !IsExpanded(m_current))
 | 
						|
            {
 | 
						|
                Expand(m_current);
 | 
						|
            }
 | 
						|
            break;
 | 
						|
#endif // __WXMSW__
 | 
						|
 | 
						|
        case '*':
 | 
						|
        case WXK_MULTIPLY:
 | 
						|
            if ( !IsExpanded(m_current) )
 | 
						|
            {
 | 
						|
                // expand all
 | 
						|
                ExpandAll(m_current);
 | 
						|
                break;
 | 
						|
            }
 | 
						|
            //else: fall through to Collapse() it
 | 
						|
 | 
						|
#ifndef __WXMSW__ // mimic the standard wxTreeCtrl behaviour
 | 
						|
        case '-':
 | 
						|
        case WXK_SUBTRACT:
 | 
						|
            if (IsExpanded(m_current))
 | 
						|
            {
 | 
						|
                Collapse(m_current);
 | 
						|
            }
 | 
						|
            break;
 | 
						|
#endif // __WXMSW__
 | 
						|
 | 
						|
        case ' ':
 | 
						|
        case WXK_RETURN:
 | 
						|
            {
 | 
						|
                wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_ACTIVATED,
 | 
						|
                                   m_owner->GetId() );
 | 
						|
                event.SetItem( (long) m_current);
 | 
						|
                event.SetEventObject( /*this*/m_owner );
 | 
						|
                m_owner->GetEventHandler()->ProcessEvent( event );
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
            // up goes to the previous sibling or to the last
 | 
						|
            // of its children if it's expanded
 | 
						|
        case WXK_UP:
 | 
						|
            {
 | 
						|
                wxTreeItemId prev = GetPrevSibling( m_key_current );
 | 
						|
                if (!prev)
 | 
						|
                {
 | 
						|
                    prev = GetParent( m_key_current );
 | 
						|
                    if ((prev == GetRootItem()) && HasFlag(wxTR_HIDE_ROOT))
 | 
						|
                    {
 | 
						|
                        break;  // don't go to root if it is hidden
 | 
						|
                    }
 | 
						|
                    if (prev)
 | 
						|
                    {
 | 
						|
                        long cookie = 0;
 | 
						|
                        wxTreeItemId current = m_key_current;
 | 
						|
                        // TODO: Huh?  If we get here, we'd better be the first child of our parent.  How else could it be?
 | 
						|
                        if (current == GetFirstChild( prev, cookie ))
 | 
						|
                        {
 | 
						|
                            // otherwise we return to where we came from
 | 
						|
                            SelectItem( prev, unselect_others, extended_select );
 | 
						|
                            m_key_current= (wxTreeListItem*) prev.m_pItem;
 | 
						|
                            EnsureVisible( prev );
 | 
						|
                            break;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                }
 | 
						|
                if (prev)
 | 
						|
                {
 | 
						|
                    while ( IsExpanded(prev) && HasChildren(prev) )
 | 
						|
                    {
 | 
						|
                        wxTreeItemId child = GetLastChild(prev);
 | 
						|
                        if ( child )
 | 
						|
                        {
 | 
						|
                            prev = child;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
 | 
						|
                    SelectItem( prev, unselect_others, extended_select );
 | 
						|
                    m_key_current=(wxTreeListItem*) prev.m_pItem;
 | 
						|
                    EnsureVisible( prev );
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
            // left arrow goes to the parent
 | 
						|
        case WXK_LEFT:
 | 
						|
#if defined(__WXMSW__) // mimic the standard win32 tree ctrl
 | 
						|
            if (IsExpanded(m_current))
 | 
						|
            {
 | 
						|
                Collapse(m_current);
 | 
						|
            }
 | 
						|
            else
 | 
						|
#endif // __WXMSW__
 | 
						|
            {
 | 
						|
                wxTreeItemId prev = GetParent( m_current );
 | 
						|
                if ((prev == GetRootItem()) && HasFlag(wxTR_HIDE_ROOT))
 | 
						|
                {
 | 
						|
                    // don't go to root if it is hidden
 | 
						|
                    prev = GetPrevSibling( m_current );
 | 
						|
                }
 | 
						|
                if (prev)
 | 
						|
                {
 | 
						|
                    EnsureVisible( prev );
 | 
						|
                    SelectItem( prev, unselect_others, extended_select );
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
        case WXK_RIGHT:
 | 
						|
#if defined(__WXMSW__) // mimic the standard win32 tree ctrl
 | 
						|
            if (m_current->HasPlus() && !IsExpanded(m_current))
 | 
						|
            {
 | 
						|
                Expand(m_current);
 | 
						|
                break;
 | 
						|
            }
 | 
						|
#endif // __WXMSW__
 | 
						|
 | 
						|
            // this works the same as the down arrow except that we
 | 
						|
            // also expand the item if it wasn't expanded yet
 | 
						|
            Expand(m_current);
 | 
						|
            // fall through
 | 
						|
 | 
						|
        case WXK_DOWN:
 | 
						|
            {
 | 
						|
                if (IsExpanded(m_key_current) && HasChildren(m_key_current))
 | 
						|
                {
 | 
						|
                    long cookie = 0;
 | 
						|
                    wxTreeItemId child = GetFirstChild( m_key_current, cookie );
 | 
						|
                    SelectItem( child, unselect_others, extended_select );
 | 
						|
                    m_key_current=(wxTreeListItem*) child.m_pItem;
 | 
						|
                    EnsureVisible( child );
 | 
						|
                }
 | 
						|
                else
 | 
						|
                {
 | 
						|
                    wxTreeItemId next = GetNextSibling( m_key_current );
 | 
						|
                    if (!next)
 | 
						|
                    {
 | 
						|
                        wxTreeItemId current = m_key_current;
 | 
						|
                        while (current && !next)
 | 
						|
                        {
 | 
						|
                            current = GetParent( current );
 | 
						|
                            if (current) next = GetNextSibling( current );
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                    if (next)
 | 
						|
                    {
 | 
						|
                        SelectItem( next, unselect_others, extended_select );
 | 
						|
                        m_key_current=(wxTreeListItem*) next.m_pItem;
 | 
						|
                        EnsureVisible( next );
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
            // <End> selects the last visible tree item
 | 
						|
        case WXK_END:
 | 
						|
            {
 | 
						|
                wxTreeItemId last = GetRootItem();
 | 
						|
 | 
						|
                while ( last.IsOk() && IsExpanded(last) )
 | 
						|
                {
 | 
						|
                    wxTreeItemId lastChild = GetLastChild(last);
 | 
						|
 | 
						|
                    // it may happen if the item was expanded but then all of
 | 
						|
                    // its children have been deleted - so IsExpanded() returned
 | 
						|
                    // TRUE, but GetLastChild() returned invalid item
 | 
						|
                    if ( !lastChild )
 | 
						|
                        break;
 | 
						|
 | 
						|
                    last = lastChild;
 | 
						|
                }
 | 
						|
 | 
						|
                if ( last.IsOk() )
 | 
						|
                {
 | 
						|
                    EnsureVisible( last );
 | 
						|
                    SelectItem( last, unselect_others, extended_select );
 | 
						|
                }
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
            // <Home> selects the root item
 | 
						|
        case WXK_HOME:
 | 
						|
            {
 | 
						|
                wxTreeItemId prev = GetRootItem();
 | 
						|
                if (!prev) break;
 | 
						|
                if (HasFlag(wxTR_HIDE_ROOT))
 | 
						|
                {
 | 
						|
                    long dummy;
 | 
						|
                    prev = GetFirstChild(prev, dummy);
 | 
						|
                    if (!prev) break;
 | 
						|
                }
 | 
						|
                EnsureVisible( prev );
 | 
						|
                SelectItem( prev, unselect_others, extended_select );
 | 
						|
            }
 | 
						|
            break;
 | 
						|
 | 
						|
        default:
 | 
						|
            event.Skip();
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
wxTreeItemId wxTreeListMainWindow::HitTest(const wxPoint& point, int& flags,
 | 
						|
                                           int& column)
 | 
						|
{
 | 
						|
    // JACS: removed wxYieldIfNeeded() because it can cause the window
 | 
						|
    // to be deleted from under us if a close window event is pending
 | 
						|
 | 
						|
    int w, h;
 | 
						|
    GetSize(&w, &h);
 | 
						|
    flags=0;
 | 
						|
    column = -1;
 | 
						|
    if (point.x<0) flags |= wxTREE_HITTEST_TOLEFT;
 | 
						|
    if (point.x>w) flags |= wxTREE_HITTEST_TORIGHT;
 | 
						|
    if (point.y<0) flags |= wxTREE_HITTEST_ABOVE;
 | 
						|
    if (point.y>h) flags |= wxTREE_HITTEST_BELOW;
 | 
						|
    if (flags) return wxTreeItemId();
 | 
						|
 | 
						|
    if (m_anchor == NULL)
 | 
						|
    {
 | 
						|
        flags = wxTREE_HITTEST_NOWHERE;
 | 
						|
        return wxTreeItemId();
 | 
						|
    }
 | 
						|
 | 
						|
    wxTreeListItem *hit =  m_anchor->HitTest(CalcUnscrolledPosition(point),
 | 
						|
                                             this, flags, column, 0);
 | 
						|
    if (hit == NULL)
 | 
						|
    {
 | 
						|
        flags = wxTREE_HITTEST_NOWHERE;
 | 
						|
        return wxTreeItemId();
 | 
						|
    }
 | 
						|
    return hit;
 | 
						|
}
 | 
						|
 | 
						|
// get the bounding rectangle of the item (or of its label only)
 | 
						|
bool wxTreeListMainWindow::GetBoundingRect(const wxTreeItemId& item,
 | 
						|
                         wxRect& rect,
 | 
						|
                         bool WXUNUSED(textOnly)) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), FALSE, _T("invalid item in wxTreeListMainWindow::GetBoundingRect") );
 | 
						|
 | 
						|
    wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    int startX, startY;
 | 
						|
    GetViewStart(& startX, & startY);
 | 
						|
 | 
						|
    rect.x = i->GetX() - startX*PIXELS_PER_UNIT;
 | 
						|
    rect.y = i->GetY() - startY*PIXELS_PER_UNIT;
 | 
						|
    rect.width = i->GetWidth();
 | 
						|
    //rect.height = i->GetHeight();
 | 
						|
    rect.height = GetLineHeight(i);
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
/* **** */
 | 
						|
 | 
						|
void wxTreeListMainWindow::Edit( const wxTreeItemId& item )
 | 
						|
{
 | 
						|
    if (!item.IsOk()) return;
 | 
						|
 | 
						|
    m_currentEdit = (wxTreeListItem*) item.m_pItem;
 | 
						|
 | 
						|
    wxTreeEvent te( wxEVT_COMMAND_TREE_BEGIN_LABEL_EDIT, m_owner->GetId() );
 | 
						|
    te.SetItem( (long) m_currentEdit);
 | 
						|
    te.SetEventObject( /*this*/m_owner );
 | 
						|
    m_owner->GetEventHandler()->ProcessEvent( te );
 | 
						|
 | 
						|
    if (!te.IsAllowed()) return;
 | 
						|
 | 
						|
    // We have to call this here because the label in
 | 
						|
    // question might just have been added and no screen
 | 
						|
    // update taken place.
 | 
						|
    if (m_dirty) wxYieldIfNeeded();
 | 
						|
 | 
						|
    wxString s = m_currentEdit->GetText(/*ALB*/m_main_column);
 | 
						|
    int x = m_currentEdit->GetX();
 | 
						|
    int y = m_currentEdit->GetY();
 | 
						|
    int w = m_currentEdit->GetWidth();
 | 
						|
    int h = m_currentEdit->GetHeight();
 | 
						|
 | 
						|
    int image_h = 0;
 | 
						|
    int image_w = 0;
 | 
						|
 | 
						|
    int image = m_currentEdit->GetCurrentImage();
 | 
						|
    if ( image != NO_IMAGE )
 | 
						|
    {
 | 
						|
        if ( m_imageListNormal )
 | 
						|
        {
 | 
						|
            m_imageListNormal->GetSize( image, image_w, image_h );
 | 
						|
            image_w += 4;
 | 
						|
        }
 | 
						|
        else
 | 
						|
        {
 | 
						|
            wxFAIL_MSG(_T("you must create an image list to use images!"));
 | 
						|
        }
 | 
						|
    }
 | 
						|
    x += image_w;
 | 
						|
    w -= image_w + 4; // I don't know why +4 is needed
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC( dc );
 | 
						|
    x = dc.LogicalToDeviceX( x );
 | 
						|
    y = dc.LogicalToDeviceY( y );
 | 
						|
 | 
						|
    wxTreeListTextCtrl *text = new wxTreeListTextCtrl(this, -1,
 | 
						|
                                              &m_renameAccept,
 | 
						|
                                              &m_renameRes,
 | 
						|
                                              this,
 | 
						|
                                              s,
 | 
						|
                                              wxPoint(x-4,y-4),
 | 
						|
                                              wxSize(w+11,h+8));
 | 
						|
    text->SetFocus();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnRenameTimer()
 | 
						|
{
 | 
						|
    Edit( m_current );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnRenameAccept()
 | 
						|
{
 | 
						|
    // TODO if the validator fails this causes a crash
 | 
						|
    wxTreeEvent le( wxEVT_COMMAND_TREE_END_LABEL_EDIT, m_owner->GetId() );
 | 
						|
    le.SetItem( (long) m_currentEdit );
 | 
						|
    le.SetEventObject( /*this*/m_owner );
 | 
						|
    le.SetLabel( m_renameRes );
 | 
						|
    m_owner->GetEventHandler()->ProcessEvent( le );
 | 
						|
 | 
						|
    if (!le.IsAllowed()) return;
 | 
						|
 | 
						|
    SetItemText( m_currentEdit, m_renameRes );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnMouse( wxMouseEvent &event )
 | 
						|
{
 | 
						|
    if ( !m_anchor ) return;
 | 
						|
 | 
						|
    // we process left mouse up event (enables in-place edit), right down
 | 
						|
    // (pass to the user code), left dbl click (activate item) and
 | 
						|
    // dragging/moving events for items drag-and-drop
 | 
						|
    if ( !(event.LeftDown() ||
 | 
						|
           event.LeftUp() ||
 | 
						|
           event.RightDown() ||
 | 
						|
           event.LeftDClick() ||
 | 
						|
           event.Dragging() ||
 | 
						|
           ((event.Moving() || event.RightUp()) && m_isDragging)) )
 | 
						|
    {
 | 
						|
        event.Skip();
 | 
						|
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    if ( event.LeftDown() )
 | 
						|
        SetFocus();
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC(dc);
 | 
						|
    wxCoord x = dc.DeviceToLogicalX( event.GetX() );
 | 
						|
    wxCoord y = dc.DeviceToLogicalY( event.GetY() );
 | 
						|
 | 
						|
    int flags = 0;
 | 
						|
    wxTreeListItem *item = m_anchor->HitTest(wxPoint(x,y), this, flags, 0);
 | 
						|
 | 
						|
    if ( event.Dragging() && !m_isDragging )
 | 
						|
    {
 | 
						|
        if (m_dragCount == 0)
 | 
						|
            m_dragStart = wxPoint(x,y);
 | 
						|
 | 
						|
        m_dragCount++;
 | 
						|
 | 
						|
        if (m_dragCount != 3)
 | 
						|
        {
 | 
						|
            // wait until user drags a bit further...
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        wxEventType command = event.RightIsDown()
 | 
						|
                              ? wxEVT_COMMAND_TREE_BEGIN_RDRAG
 | 
						|
                              : wxEVT_COMMAND_TREE_BEGIN_DRAG;
 | 
						|
 | 
						|
        wxTreeEvent nevent( command,/*ALB*/ m_owner->GetId() );
 | 
						|
        nevent.SetItem( (long) m_current);
 | 
						|
        nevent.SetEventObject(/*this*/m_owner); // ALB
 | 
						|
 | 
						|
        // by default the dragging is not supported, the user code must
 | 
						|
        // explicitly allow the event for it to take place
 | 
						|
        nevent.Veto();
 | 
						|
 | 
						|
        if ( m_owner->GetEventHandler()->ProcessEvent(nevent) &&
 | 
						|
             nevent.IsAllowed() )
 | 
						|
        {
 | 
						|
            // we're going to drag this item
 | 
						|
            m_isDragging = TRUE;
 | 
						|
 | 
						|
            // remember the old cursor because we will change it while
 | 
						|
            // dragging
 | 
						|
            m_oldCursor = m_cursor;
 | 
						|
 | 
						|
            // in a single selection control, hide the selection temporarily
 | 
						|
            if ( !(GetWindowStyleFlag() & wxTR_MULTIPLE) )
 | 
						|
            {
 | 
						|
                m_oldSelection = (wxTreeListItem*) GetSelection().m_pItem;
 | 
						|
 | 
						|
                if ( m_oldSelection )
 | 
						|
                {
 | 
						|
                    m_oldSelection->SetHilight(FALSE);
 | 
						|
                    RefreshLine(m_oldSelection);
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            CaptureMouse();
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else if ( event.Moving() )
 | 
						|
    {
 | 
						|
        if ( item != m_dropTarget )
 | 
						|
        {
 | 
						|
            // unhighlight the previous drop target
 | 
						|
            DrawDropEffect(m_dropTarget);
 | 
						|
 | 
						|
            m_dropTarget = item;
 | 
						|
 | 
						|
            // highlight the current drop target if any
 | 
						|
            DrawDropEffect(m_dropTarget);
 | 
						|
 | 
						|
            wxYieldIfNeeded();
 | 
						|
        }
 | 
						|
    }
 | 
						|
    else if ( (event.LeftUp() || event.RightUp()) && m_isDragging )
 | 
						|
    {
 | 
						|
        // erase the highlighting
 | 
						|
        DrawDropEffect(m_dropTarget);
 | 
						|
 | 
						|
        if ( m_oldSelection )
 | 
						|
        {
 | 
						|
            m_oldSelection->SetHilight(TRUE);
 | 
						|
            RefreshLine(m_oldSelection);
 | 
						|
            m_oldSelection = (wxTreeListItem *)NULL;
 | 
						|
        }
 | 
						|
 | 
						|
        // generate the drag end event
 | 
						|
        wxTreeEvent event(wxEVT_COMMAND_TREE_END_DRAG,/*ALB*/m_owner->GetId());
 | 
						|
 | 
						|
        event.SetItem( (long) item );
 | 
						|
        event.SetPoint( wxPoint(x, y) );
 | 
						|
        event.SetEventObject(/*this*/m_owner);
 | 
						|
 | 
						|
        (void)m_owner->GetEventHandler()->ProcessEvent(event);
 | 
						|
 | 
						|
        m_isDragging = FALSE;
 | 
						|
        m_dropTarget = (wxTreeListItem *)NULL;
 | 
						|
 | 
						|
        ReleaseMouse();
 | 
						|
 | 
						|
        SetCursor(m_oldCursor);
 | 
						|
 | 
						|
        wxYieldIfNeeded();
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        // here we process only the messages which happen on tree items
 | 
						|
 | 
						|
        m_dragCount = 0;
 | 
						|
 | 
						|
        if (item == NULL) return;  /* we hit the blank area */
 | 
						|
 | 
						|
        if ( event.RightDown() )
 | 
						|
        {
 | 
						|
            SetFocus();
 | 
						|
            wxTreeEvent nevent(wxEVT_COMMAND_TREE_ITEM_RIGHT_CLICK,
 | 
						|
                               m_owner->GetId());
 | 
						|
            nevent.SetItem( (long) item );
 | 
						|
            int nx, ny;
 | 
						|
            CalcScrolledPosition(x, y, &nx, &ny);
 | 
						|
            nevent.SetPoint( wxPoint(nx, ny));
 | 
						|
            nevent.SetEventObject(/*this*/m_owner);
 | 
						|
            m_owner->GetEventHandler()->ProcessEvent(nevent);
 | 
						|
        }
 | 
						|
        else if ( event.LeftUp() )
 | 
						|
        {
 | 
						|
            if ( m_lastOnSame )
 | 
						|
            {
 | 
						|
                if ( (item == m_current) &&
 | 
						|
                     (flags & wxTREE_HITTEST_ONITEMLABEL) &&
 | 
						|
                     HasFlag(wxTR_EDIT_LABELS) )
 | 
						|
                {
 | 
						|
                    if ( m_renameTimer->IsRunning() )
 | 
						|
                        m_renameTimer->Stop();
 | 
						|
 | 
						|
                    m_renameTimer->Start( 100, TRUE );
 | 
						|
                }
 | 
						|
 | 
						|
                m_lastOnSame = FALSE;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else // !RightDown() && !LeftUp() ==> LeftDown() || LeftDClick()
 | 
						|
        {
 | 
						|
            if ( event.LeftDown() )
 | 
						|
            {
 | 
						|
                SetFocus();
 | 
						|
                m_lastOnSame = item == m_current;
 | 
						|
            }
 | 
						|
 | 
						|
            if ( flags & wxTREE_HITTEST_ONITEMBUTTON )
 | 
						|
            {
 | 
						|
                // only toggle the item for a single click, double click on
 | 
						|
                // the button doesn't do anything (it toggles the item twice)
 | 
						|
                if ( event.LeftDown() )
 | 
						|
                {
 | 
						|
                    Toggle( item );
 | 
						|
                }
 | 
						|
 | 
						|
                // don't select the item if the button was clicked
 | 
						|
                return;
 | 
						|
            }
 | 
						|
 | 
						|
            // how should the selection work for this event?
 | 
						|
            bool is_multiple, extended_select, unselect_others;
 | 
						|
            EventFlagsToSelType(GetWindowStyleFlag(),
 | 
						|
                                event.ShiftDown(),
 | 
						|
                                event.ControlDown(),
 | 
						|
                                is_multiple, extended_select, unselect_others);
 | 
						|
 | 
						|
            SelectItem(item, unselect_others, extended_select);
 | 
						|
 | 
						|
            // For some reason, Windows isn't recognizing a left double-click,
 | 
						|
            // so we need to simulate it here.  Allow 200 milliseconds for now.
 | 
						|
            if ( event.LeftDClick() )
 | 
						|
            {
 | 
						|
                // double clicking should not start editing the item label
 | 
						|
                m_renameTimer->Stop();
 | 
						|
                m_lastOnSame = FALSE;
 | 
						|
 | 
						|
                // send activate event first
 | 
						|
                wxTreeEvent nevent( wxEVT_COMMAND_TREE_ITEM_ACTIVATED,
 | 
						|
                                    m_owner->GetId() );
 | 
						|
                nevent.SetItem( (long) item );
 | 
						|
                int nx, ny;
 | 
						|
                CalcScrolledPosition(x, y, &nx, &ny);
 | 
						|
                nevent.SetPoint( wxPoint(nx, ny) );
 | 
						|
                nevent.SetEventObject( /*this*/m_owner );
 | 
						|
                if ( !m_owner->GetEventHandler()->ProcessEvent( nevent ) )
 | 
						|
                {
 | 
						|
                    // if the user code didn't process the activate event,
 | 
						|
                    // handle it ourselves by toggling the item when it is
 | 
						|
                    // double clicked
 | 
						|
                    if ( item->HasPlus() )
 | 
						|
                    {
 | 
						|
                        Toggle(item);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnIdle( wxIdleEvent &WXUNUSED(event) )
 | 
						|
{
 | 
						|
    /* after all changes have been done to the tree control,
 | 
						|
     * we actually redraw the tree when everything is over */
 | 
						|
 | 
						|
    if (!m_dirty) return;
 | 
						|
 | 
						|
    m_dirty = FALSE;
 | 
						|
 | 
						|
    CalculatePositions();
 | 
						|
    Refresh();
 | 
						|
    AdjustMyScrollbars();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnSize(wxSizeEvent& WXUNUSED(event))
 | 
						|
{
 | 
						|
//     int w, h;
 | 
						|
//     GetClientSize(&w, &h);
 | 
						|
//     m_header_win->SetSize(0, 0, w, HEADER_HEIGHT);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::OnScroll(wxScrollWinEvent& event)
 | 
						|
{
 | 
						|
    // FIXME
 | 
						|
#if defined(__WXGTK__) && !defined(__WXUNIVERSAL__)
 | 
						|
    wxScrolledWindow::OnScroll(event);
 | 
						|
#else
 | 
						|
    HandleOnScroll( event );
 | 
						|
#endif
 | 
						|
 | 
						|
    if(event.GetOrientation() == wxHORIZONTAL)
 | 
						|
    {
 | 
						|
        m_owner->GetHeaderWindow()->Refresh();
 | 
						|
#ifdef __WXMAC__
 | 
						|
        m_owner->GetHeaderWindow()->MacUpdateImmediately();
 | 
						|
#endif
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
void wxTreeListMainWindow::CalculateSize( wxTreeListItem *item, wxDC &dc )
 | 
						|
{
 | 
						|
    wxCoord text_w = 0;
 | 
						|
    wxCoord text_h = 0;
 | 
						|
 | 
						|
    if (item->IsBold())
 | 
						|
        dc.SetFont(m_boldFont);
 | 
						|
 | 
						|
    dc.GetTextExtent( item->GetText(/*ALB*/m_main_column), &text_w, &text_h );
 | 
						|
    text_h+=2;
 | 
						|
 | 
						|
    // restore normal font
 | 
						|
    dc.SetFont( m_normalFont );
 | 
						|
 | 
						|
    int image_h = 0;
 | 
						|
    int image_w = 0;
 | 
						|
    int image = item->GetCurrentImage();
 | 
						|
    if ( image != NO_IMAGE )
 | 
						|
    {
 | 
						|
        if ( m_imageListNormal )
 | 
						|
        {
 | 
						|
            m_imageListNormal->GetSize( image, image_w, image_h );
 | 
						|
            image_w += 4;
 | 
						|
            image_h += 2;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    int total_h = (image_h > text_h) ? image_h : text_h;
 | 
						|
 | 
						|
//     if (total_h < 30)
 | 
						|
//         total_h += 2;            // at least 2 pixels
 | 
						|
//     else
 | 
						|
//         total_h += total_h/10;   // otherwise 10% extra spacing
 | 
						|
 | 
						|
    item->SetHeight(total_h);
 | 
						|
    if (total_h>m_lineHeight)
 | 
						|
        m_lineHeight=total_h;
 | 
						|
 | 
						|
    item->SetWidth(image_w+text_w+2);
 | 
						|
}
 | 
						|
 | 
						|
// -----------------------------------------------------------------------------
 | 
						|
// for developper : y is now the top of the level
 | 
						|
// not the middle of it !
 | 
						|
void wxTreeListMainWindow::CalculateLevel( wxTreeListItem *item, wxDC &dc,
 | 
						|
                                        int level, int &y, int x_offset )
 | 
						|
{
 | 
						|
    int x = level*m_indent + x_offset;
 | 
						|
    if (!HasFlag(wxTR_HIDE_ROOT))
 | 
						|
    {
 | 
						|
        x += m_indent;
 | 
						|
    }
 | 
						|
    else if (level == 0)
 | 
						|
    {
 | 
						|
        // a hidden root is not evaluated, but its
 | 
						|
        // children are always calculated
 | 
						|
        goto Recurse;
 | 
						|
    }
 | 
						|
 | 
						|
    CalculateSize( item, dc );
 | 
						|
 | 
						|
    // set its position
 | 
						|
    item->SetX( x+m_spacing );
 | 
						|
    item->SetY( y );
 | 
						|
    y += GetLineHeight(item);
 | 
						|
 | 
						|
    if ( !item->IsExpanded() )
 | 
						|
    {
 | 
						|
        // we don't need to calculate collapsed branches
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
  Recurse:
 | 
						|
    wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
    size_t n, count = children.Count();
 | 
						|
    ++level;
 | 
						|
    for (n = 0; n < count; ++n )
 | 
						|
        CalculateLevel( children[n], dc, level, y, x_offset );  // recurse
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::CalculatePositions()
 | 
						|
{
 | 
						|
    if ( !m_anchor ) return;
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC( dc );
 | 
						|
 | 
						|
    dc.SetFont( m_normalFont );
 | 
						|
 | 
						|
    dc.SetPen( m_dottedPen );
 | 
						|
    //if(GetImageList() == NULL)
 | 
						|
    // m_lineHeight = (int)(dc.GetCharHeight() + 4);
 | 
						|
 | 
						|
    int y = 2;
 | 
						|
    int x_offset = 0;
 | 
						|
    for(size_t i = 0; i < GetMainColumn(); ++i) {
 | 
						|
        x_offset += m_owner->GetHeaderWindow()->GetColumnWidth(i);
 | 
						|
    }
 | 
						|
    CalculateLevel( m_anchor, dc, 0, y, x_offset ); // start recursion
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::RefreshSubtree(wxTreeListItem *item)
 | 
						|
{
 | 
						|
    if (m_dirty) return;
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC(dc);
 | 
						|
 | 
						|
    int cw = 0;
 | 
						|
    int ch = 0;
 | 
						|
    //GetClientSize( &cw, &ch );
 | 
						|
    GetVirtualSize(&cw, &ch);
 | 
						|
 | 
						|
    wxRect rect;
 | 
						|
    rect.x = dc.LogicalToDeviceX( 0 );
 | 
						|
    rect.width = cw;
 | 
						|
    rect.y = dc.LogicalToDeviceY( item->GetY() - 2 );
 | 
						|
    rect.height = ch;
 | 
						|
 | 
						|
    Refresh( TRUE, &rect );
 | 
						|
 | 
						|
    AdjustMyScrollbars();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::RefreshLine( wxTreeListItem *item )
 | 
						|
{
 | 
						|
    if (m_dirty) return;
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    PrepareDC( dc );
 | 
						|
 | 
						|
    int cw = 0;
 | 
						|
    int ch = 0;
 | 
						|
    //GetClientSize( &cw, &ch );
 | 
						|
    GetVirtualSize(&cw, &ch);
 | 
						|
 | 
						|
    wxRect rect;
 | 
						|
    rect.x = dc.LogicalToDeviceX( 0 );
 | 
						|
    rect.y = dc.LogicalToDeviceY( item->GetY() );
 | 
						|
    rect.width = cw;
 | 
						|
    rect.height = GetLineHeight(item); //dc.GetCharHeight() + 6;
 | 
						|
 | 
						|
    Refresh( TRUE, &rect );
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::RefreshSelected()
 | 
						|
{
 | 
						|
    // TODO: this is awfully inefficient, we should keep the list of all
 | 
						|
    //       selected items internally, should be much faster
 | 
						|
    if ( m_anchor )
 | 
						|
        RefreshSelectedUnder(m_anchor);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListMainWindow::RefreshSelectedUnder(wxTreeListItem *item)
 | 
						|
{
 | 
						|
    if ( item->IsSelected() )
 | 
						|
        RefreshLine(item);
 | 
						|
 | 
						|
    const wxArrayTreeListItems& children = item->GetChildren();
 | 
						|
    size_t count = children.GetCount();
 | 
						|
    for ( size_t n = 0; n < count; n++ )
 | 
						|
    {
 | 
						|
        RefreshSelectedUnder(children[n]);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
// changing colours: we need to refresh the tree control
 | 
						|
// ----------------------------------------------------------------------------
 | 
						|
 | 
						|
bool wxTreeListMainWindow::SetBackgroundColour(const wxColour& colour)
 | 
						|
{
 | 
						|
    if ( !wxWindow::SetBackgroundColour(colour) )
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    Refresh();
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
bool wxTreeListMainWindow::SetForegroundColour(const wxColour& colour)
 | 
						|
{
 | 
						|
    if ( !wxWindow::SetForegroundColour(colour) )
 | 
						|
        return FALSE;
 | 
						|
 | 
						|
    Refresh();
 | 
						|
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
//----------- ALB -------------
 | 
						|
inline
 | 
						|
void wxTreeListMainWindow::SetItemText(const wxTreeItemId& item, size_t column,
 | 
						|
                                    const wxString& text)
 | 
						|
{
 | 
						|
    wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
 | 
						|
 | 
						|
    wxClientDC dc(this);
 | 
						|
    wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
 | 
						|
    pItem->SetText(column, text);
 | 
						|
    CalculateSize(pItem, dc);
 | 
						|
    RefreshLine(pItem);
 | 
						|
}
 | 
						|
 | 
						|
inline
 | 
						|
wxString wxTreeListMainWindow::GetItemText(const wxTreeItemId& item,
 | 
						|
                                     size_t column) const
 | 
						|
{
 | 
						|
    wxCHECK_MSG( item.IsOk(), wxT(""), wxT("invalid tree item") );
 | 
						|
 | 
						|
    return ((wxTreeListItem*) item.m_pItem)->GetText(column);
 | 
						|
}
 | 
						|
 | 
						|
//-----------------------------
 | 
						|
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
//  wxTreeListCtrl
 | 
						|
//-----------------------------------------------------------------------------
 | 
						|
 | 
						|
IMPLEMENT_DYNAMIC_CLASS(wxTreeListCtrl, wxControl);
 | 
						|
 | 
						|
BEGIN_EVENT_TABLE(wxTreeListCtrl, wxControl)
 | 
						|
    EVT_SIZE(wxTreeListCtrl::OnSize)
 | 
						|
END_EVENT_TABLE();
 | 
						|
 | 
						|
bool wxTreeListCtrl::Create(wxWindow *parent, wxWindowID id,
 | 
						|
                            const wxPoint& pos,
 | 
						|
                            const wxSize& size,
 | 
						|
                            long style, const wxValidator &validator,
 | 
						|
                            const wxString& name)
 | 
						|
{
 | 
						|
    long main_style = style & ~(wxRAISED_BORDER|wxSUNKEN_BORDER
 | 
						|
                                |wxSIMPLE_BORDER|wxNO_BORDER|wxDOUBLE_BORDER
 | 
						|
                                |wxSTATIC_BORDER);
 | 
						|
    if(!wxControl::Create(parent, id, pos, size, style, validator, name))
 | 
						|
        return false;
 | 
						|
 | 
						|
    m_main_win = new wxTreeListMainWindow(this, -1, wxPoint(0, 0), size,
 | 
						|
                                          main_style, validator);
 | 
						|
    m_header_win = new wxTreeListHeaderWindow(this, -1, m_main_win,
 | 
						|
                                              wxPoint(0, 0), wxDefaultSize,
 | 
						|
                                              wxTAB_TRAVERSAL);
 | 
						|
    return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListCtrl::OnSize(wxSizeEvent& event)
 | 
						|
{
 | 
						|
    int w, h;
 | 
						|
    GetClientSize(&w, &h);
 | 
						|
    if(m_header_win)
 | 
						|
        m_header_win->SetSize(0, 0, w, HEADER_HEIGHT);
 | 
						|
    if(m_main_win)
 | 
						|
        m_main_win->SetSize(0, HEADER_HEIGHT + 1, w, h - HEADER_HEIGHT - 1);
 | 
						|
}
 | 
						|
 | 
						|
size_t wxTreeListCtrl::GetCount() const { return m_main_win->GetCount(); }
 | 
						|
 | 
						|
unsigned int wxTreeListCtrl::GetIndent() const
 | 
						|
{ return m_main_win->GetIndent(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetIndent(unsigned int indent)
 | 
						|
{ m_main_win->SetIndent(indent); }
 | 
						|
 | 
						|
unsigned int wxTreeListCtrl::GetSpacing() const
 | 
						|
{ return m_main_win->GetSpacing(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetSpacing(unsigned int spacing)
 | 
						|
{ m_main_win->SetSpacing(spacing); }
 | 
						|
 | 
						|
unsigned int wxTreeListCtrl::GetLineSpacing() const
 | 
						|
{ return m_main_win->GetLineSpacing(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetLineSpacing(unsigned int spacing)
 | 
						|
{ m_main_win->SetLineSpacing(spacing); }
 | 
						|
 | 
						|
wxImageList* wxTreeListCtrl::GetImageList() const
 | 
						|
{ return m_main_win->GetImageList(); }
 | 
						|
 | 
						|
wxImageList* wxTreeListCtrl::GetStateImageList() const
 | 
						|
{ return m_main_win->GetStateImageList(); }
 | 
						|
 | 
						|
wxImageList* wxTreeListCtrl::GetButtonsImageList() const
 | 
						|
{ return m_main_win->GetButtonsImageList(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetImageList(wxImageList* imageList)
 | 
						|
{ m_main_win->SetImageList(imageList); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetStateImageList(wxImageList* imageList)
 | 
						|
{ m_main_win->SetStateImageList(imageList); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetButtonsImageList(wxImageList* imageList)
 | 
						|
{ m_main_win->SetButtonsImageList(imageList); }
 | 
						|
 | 
						|
void wxTreeListCtrl::AssignImageList(wxImageList* imageList)
 | 
						|
{ m_main_win->AssignImageList(imageList); }
 | 
						|
 | 
						|
void wxTreeListCtrl::AssignStateImageList(wxImageList* imageList)
 | 
						|
{ m_main_win->AssignStateImageList(imageList); }
 | 
						|
 | 
						|
void wxTreeListCtrl::AssignButtonsImageList(wxImageList* imageList)
 | 
						|
{ m_main_win->AssignButtonsImageList(imageList); }
 | 
						|
 | 
						|
wxString wxTreeListCtrl::GetItemText(const wxTreeItemId& item, size_t column)
 | 
						|
    const
 | 
						|
{ return m_main_win->GetItemText(item, column); }
 | 
						|
 | 
						|
int wxTreeListCtrl::GetItemImage(const wxTreeItemId& item, size_t column,
 | 
						|
                                 wxTreeItemIcon which) const
 | 
						|
{ return m_main_win->GetItemImage(item, column, which); }
 | 
						|
 | 
						|
wxTreeItemData* wxTreeListCtrl::GetItemData(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetItemData(item); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::GetItemBold(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetItemBold(item); }
 | 
						|
 | 
						|
wxColour wxTreeListCtrl::GetItemTextColour(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetItemTextColour(item); }
 | 
						|
 | 
						|
wxColour wxTreeListCtrl::GetItemBackgroundColour(const wxTreeItemId& item)
 | 
						|
    const
 | 
						|
{ return m_main_win->GetItemBackgroundColour(item); }
 | 
						|
 | 
						|
wxFont wxTreeListCtrl::GetItemFont(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetItemFont(item); }
 | 
						|
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemText(const wxTreeItemId& item, size_t column,
 | 
						|
                                 const wxString& text)
 | 
						|
{ m_main_win->SetItemText(item, column, text); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemImage(const wxTreeItemId& item,
 | 
						|
                                  size_t column,
 | 
						|
                                  int image,
 | 
						|
                                  wxTreeItemIcon which)
 | 
						|
{ m_main_win->SetItemImage(item, column, image, which); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemData(const wxTreeItemId& item,
 | 
						|
                                 wxTreeItemData* data)
 | 
						|
{ m_main_win->SetItemData(item, data); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemHasChildren(const wxTreeItemId& item, bool has)
 | 
						|
{ m_main_win->SetItemHasChildren(item, has); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemBold(const wxTreeItemId& item, bool bold)
 | 
						|
{ m_main_win->SetItemBold(item, bold); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemTextColour(const wxTreeItemId& item,
 | 
						|
                                       const wxColour& col)
 | 
						|
{ m_main_win->SetItemTextColour(item, col); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemBackgroundColour(const wxTreeItemId& item,
 | 
						|
                                             const wxColour& col)
 | 
						|
{ m_main_win->SetItemBackgroundColour(item, col); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetItemFont(const wxTreeItemId& item,
 | 
						|
                                 const wxFont& font)
 | 
						|
{ m_main_win->SetItemFont(item, font); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::SetFont(const wxFont& font)
 | 
						|
{
 | 
						|
    if(m_header_win) m_header_win->SetFont(font);
 | 
						|
    if(m_main_win)
 | 
						|
        return m_main_win->SetFont(font);
 | 
						|
    else return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListCtrl::SetWindowStyle(const long style)
 | 
						|
{
 | 
						|
    if(m_main_win)
 | 
						|
        m_main_win->SetWindowStyle(style);
 | 
						|
    // TODO: provide something like wxTL_NO_HEADERS to hide m_header_win
 | 
						|
}
 | 
						|
 | 
						|
long wxTreeListCtrl::GetWindowStyle() const
 | 
						|
{
 | 
						|
    long style = m_windowStyle;
 | 
						|
    if(m_main_win)
 | 
						|
        style |= m_main_win->GetWindowStyle();
 | 
						|
    return style;
 | 
						|
}
 | 
						|
 | 
						|
bool wxTreeListCtrl::IsVisible(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->IsVisible(item); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::ItemHasChildren(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->ItemHasChildren(item); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::IsExpanded(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->IsExpanded(item); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::IsSelected(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->IsSelected(item); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::IsBold(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->IsBold(item); }
 | 
						|
 | 
						|
size_t wxTreeListCtrl::GetChildrenCount(const wxTreeItemId& item, bool rec)
 | 
						|
{ return m_main_win->GetChildrenCount(item, rec); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetRootItem() const
 | 
						|
{ return m_main_win->GetRootItem(); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetSelection() const
 | 
						|
{ return m_main_win->GetSelection(); }
 | 
						|
 | 
						|
size_t wxTreeListCtrl::GetSelections(wxArrayTreeItemIds& arr) const
 | 
						|
{ return m_main_win->GetSelections(arr); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetParent(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetParent(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetFirstChild(const wxTreeItemId& item,
 | 
						|
                                           long& cookie) const
 | 
						|
{ return m_main_win->GetFirstChild(item, cookie); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetNextChild(const wxTreeItemId& item,
 | 
						|
                                          long& cookie) const
 | 
						|
{ return m_main_win->GetNextChild(item, cookie); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetLastChild(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetLastChild(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetNextSibling(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetNextSibling(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetPrevSibling(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetPrevSibling(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetFirstVisibleItem() const
 | 
						|
{ return m_main_win->GetFirstVisibleItem(); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetNextVisible(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetNextVisible(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetPrevVisible(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetPrevVisible(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::GetNext(const wxTreeItemId& item) const
 | 
						|
{ return m_main_win->GetNext(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::AddRoot(const wxString& text, int image,
 | 
						|
                                     int selectedImage, wxTreeItemData* data)
 | 
						|
{ return m_main_win->AddRoot(text, image, selectedImage, data); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::PrependItem(const wxTreeItemId& parent,
 | 
						|
                                         const wxString& text, int image,
 | 
						|
                                         int selectedImage,
 | 
						|
                                         wxTreeItemData* data)
 | 
						|
{ return m_main_win->PrependItem(parent, text, image, selectedImage, data); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::InsertItem(const wxTreeItemId& parent,
 | 
						|
                                        const wxTreeItemId& previous,
 | 
						|
                                        const wxString& text, int image,
 | 
						|
                                        int selectedImage,
 | 
						|
                                        wxTreeItemData* data)
 | 
						|
{
 | 
						|
    return m_main_win->InsertItem(parent, previous, text, image,
 | 
						|
                                  selectedImage, data);
 | 
						|
}
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::InsertItem(const wxTreeItemId& parent,
 | 
						|
                                        size_t index,
 | 
						|
                                        const wxString& text, int image,
 | 
						|
                                        int selectedImage,
 | 
						|
                                        wxTreeItemData* data)
 | 
						|
{
 | 
						|
    return m_main_win->InsertItem(parent, index, text, image,
 | 
						|
                                  selectedImage, data);
 | 
						|
}
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::AppendItem(const wxTreeItemId& parent,
 | 
						|
                                        const wxString& text, int image,
 | 
						|
                                        int selectedImage,
 | 
						|
                                        wxTreeItemData* data)
 | 
						|
{ return m_main_win->AppendItem(parent, text, image, selectedImage, data); }
 | 
						|
 | 
						|
void wxTreeListCtrl::Delete(const wxTreeItemId& item)
 | 
						|
{ m_main_win->Delete(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::DeleteChildren(const wxTreeItemId& item)
 | 
						|
{ m_main_win->DeleteChildren(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::DeleteAllItems()
 | 
						|
{ m_main_win->DeleteAllItems(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::Expand(const wxTreeItemId& item)
 | 
						|
{ m_main_win->Expand(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::ExpandAll(const wxTreeItemId& item)
 | 
						|
{ m_main_win->ExpandAll(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::Collapse(const wxTreeItemId& item)
 | 
						|
{ m_main_win->Collapse(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::CollapseAndReset(const wxTreeItemId& item)
 | 
						|
{ m_main_win->CollapseAndReset(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::Toggle(const wxTreeItemId& item)
 | 
						|
{ m_main_win->Toggle(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::Unselect()
 | 
						|
{ m_main_win->Unselect(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::UnselectAll()
 | 
						|
{ m_main_win->UnselectAll(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SelectItem(const wxTreeItemId& item, bool unselect_others,
 | 
						|
                                bool extended_select)
 | 
						|
{ m_main_win->SelectItem(item, unselect_others, extended_select); }
 | 
						|
 | 
						|
void wxTreeListCtrl::EnsureVisible(const wxTreeItemId& item)
 | 
						|
{ m_main_win->EnsureVisible(item); }
 | 
						|
 | 
						|
void wxTreeListCtrl::ScrollTo(const wxTreeItemId& item)
 | 
						|
{ m_main_win->ScrollTo(item); }
 | 
						|
 | 
						|
wxTreeItemId wxTreeListCtrl::HitTest(const wxPoint& pos, int& flags,
 | 
						|
                                     int& column)
 | 
						|
{
 | 
						|
    return m_main_win->HitTest(pos, flags, column);
 | 
						|
}
 | 
						|
 | 
						|
bool wxTreeListCtrl::GetBoundingRect(const wxTreeItemId& item, wxRect& rect,
 | 
						|
                                     bool textOnly) const
 | 
						|
{ return m_main_win->GetBoundingRect(item, rect, textOnly); }
 | 
						|
 | 
						|
void wxTreeListCtrl::Edit(const wxTreeItemId& item)
 | 
						|
{ m_main_win->Edit(item); }
 | 
						|
 | 
						|
int wxTreeListCtrl::OnCompareItems(const wxTreeItemId& item1,
 | 
						|
                                   const wxTreeItemId& item2)
 | 
						|
{
 | 
						|
    // ALB: do the comparison here, and not delegate to m_main_win, in order
 | 
						|
    // to let the user override it
 | 
						|
    //return m_main_win->OnCompareItems(item1, item2);
 | 
						|
    return wxStrcmp(GetItemText(item1), GetItemText(item2));
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListCtrl::SortChildren(const wxTreeItemId& item)
 | 
						|
{ m_main_win->SortChildren(item); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::SetBackgroundColour(const wxColour& colour)
 | 
						|
{ return m_main_win->SetBackgroundColour(colour); }
 | 
						|
 | 
						|
bool wxTreeListCtrl::SetForegroundColour(const wxColour& colour)
 | 
						|
{ return m_main_win->SetForegroundColour(colour); }
 | 
						|
 | 
						|
size_t wxTreeListCtrl::GetColumnCount() const
 | 
						|
{ return m_main_win->GetColumnCount(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetColumnWidth(size_t column, size_t width)
 | 
						|
{ m_header_win->SetColumnWidth(column, width); }
 | 
						|
 | 
						|
int wxTreeListCtrl::GetColumnWidth(size_t column) const
 | 
						|
{ return m_header_win->GetColumnWidth(column); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetMainColumn(size_t column)
 | 
						|
{ m_main_win->SetMainColumn(column); }
 | 
						|
 | 
						|
size_t wxTreeListCtrl::GetMainColumn() const
 | 
						|
{ return m_main_win->GetMainColumn(); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetColumnText(size_t column, const wxString& text)
 | 
						|
{
 | 
						|
    m_header_win->SetColumnText(column, text);
 | 
						|
    m_header_win->Refresh();
 | 
						|
}
 | 
						|
 | 
						|
wxString wxTreeListCtrl::GetColumnText(size_t column) const
 | 
						|
{ return m_header_win->GetColumnText(column); }
 | 
						|
 | 
						|
void wxTreeListCtrl::AddColumn(const wxTreeListColumnInfo& col)
 | 
						|
{ m_header_win->AddColumn(col); }
 | 
						|
 | 
						|
void wxTreeListCtrl::InsertColumn(size_t before,
 | 
						|
                                  const wxTreeListColumnInfo& col)
 | 
						|
{ m_header_win->InsertColumn(before, col); }
 | 
						|
 | 
						|
void wxTreeListCtrl::RemoveColumn(size_t column)
 | 
						|
{ m_header_win->RemoveColumn(column); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetColumn(size_t column, const wxTreeListColumnInfo& col)
 | 
						|
{ m_header_win->SetColumn(column, col); }
 | 
						|
 | 
						|
const wxTreeListColumnInfo& wxTreeListCtrl::GetColumn(size_t column) const
 | 
						|
{ return m_header_win->GetColumn(column); }
 | 
						|
 | 
						|
wxTreeListColumnInfo& wxTreeListCtrl::GetColumn(size_t column)
 | 
						|
{ return m_header_win->GetColumn(column); }
 | 
						|
 | 
						|
void wxTreeListCtrl::SetColumnImage(size_t column, int image)
 | 
						|
{
 | 
						|
    m_header_win->SetColumn(column, GetColumn(column).SetImage(image));
 | 
						|
}
 | 
						|
 | 
						|
int wxTreeListCtrl::GetColumnImage(size_t column) const
 | 
						|
{
 | 
						|
    return m_header_win->GetColumn(column).GetImage();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListCtrl::SetColumnAlignment(size_t column,
 | 
						|
                                        wxTreeListColumnAlign align)
 | 
						|
{
 | 
						|
    m_header_win->SetColumn(column, GetColumn(column).SetAlignment(align));
 | 
						|
}
 | 
						|
 | 
						|
wxTreeListColumnAlign wxTreeListCtrl::GetColumnAlignment(size_t column) const
 | 
						|
{
 | 
						|
    return m_header_win->GetColumn(column).GetAlignment();
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListCtrl::Refresh(bool erase, const wxRect* rect)
 | 
						|
{
 | 
						|
    m_main_win->Refresh(erase, rect);
 | 
						|
    m_header_win->Refresh(erase, rect);
 | 
						|
}
 | 
						|
 | 
						|
void wxTreeListCtrl::SetFocus()
 | 
						|
{ m_main_win->SetFocus(); }
 |