]> git.saurik.com Git - wxWidgets.git/blob - wxPython/contrib/gizmos/wxCode/src/treelistctrl.cpp
288ea108b64175b841ab44b70b3a24389dbe308b
[wxWidgets.git] / wxPython / contrib / gizmos / wxCode / src / treelistctrl.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: treelistctrl.cpp
3 // Purpose: multi column tree control implementation
4 // Author: Robert Roebling
5 // Created: 01/02/97
6 // Modified: Alberto Griggio, 2002
7 // 22/10/98 - almost total rewrite, simpler interface (VZ)
8 // Id: $Id$
9 // Copyright: (c) Robert Roebling, Julian Smart, Alberto Griggio,
10 // Vadim Zeitlin, Otto Wyss
11 // Licence: wxWindows licence
12 /////////////////////////////////////////////////////////////////////////////
13
14 // ===========================================================================
15 // declarations
16 // ===========================================================================
17
18 // ---------------------------------------------------------------------------
19 // headers
20 // ---------------------------------------------------------------------------
21
22 #if defined(__GNUG__) && !defined(__APPLE__)
23 #pragma implementation "treelistctrl.h"
24 #endif
25
26 // For compilers that support precompilation, includes "wx.h".
27 #include "wx/wxprec.h"
28
29 #ifdef __BORLANDC__
30 #pragma hdrstop
31 #endif
32
33
34 #include <wx/treebase.h>
35 #include <wx/timer.h>
36 #include <wx/textctrl.h>
37 #include <wx/imaglist.h>
38 #include <wx/settings.h>
39 #include <wx/dcclient.h>
40 #include <wx/dcscreen.h>
41 #include <wx/scrolwin.h>
42
43 #include "wx/treelistctrl.h"
44
45 #ifdef __WXGTK__
46 #include <gtk/gtk.h>
47 #include <wx/gtk/win_gtk.h>
48 #endif
49
50 // ---------------------------------------------------------------------------
51 // array types
52 // ---------------------------------------------------------------------------
53
54 class wxTreeListItem;
55
56 #if !wxCHECK_VERSION(2, 5, 0)
57 WX_DEFINE_ARRAY(wxTreeListItem *, wxArrayTreeListItems);
58 #else
59 WX_DEFINE_ARRAY_PTR(wxTreeListItem *, wxArrayTreeListItems);
60 #endif
61
62 #include <wx/dynarray.h>
63 WX_DECLARE_OBJARRAY(wxTreeListColumnInfo, wxArrayTreeListColumnInfo);
64 #include <wx/arrimpl.cpp>
65 WX_DEFINE_OBJARRAY(wxArrayTreeListColumnInfo);
66
67 #if !wxCHECK_VERSION(2, 3, 3)
68 WX_DEFINE_ARRAY(short, wxArrayShort);
69 #endif
70
71
72 // --------------------------------------------------------------------------
73 // constants
74 // --------------------------------------------------------------------------
75
76 static const int NO_IMAGE = -1;
77
78 const int LINEHEIGHT = 10;
79 const int PIXELS_PER_UNIT = 10;
80 const int LINEATROOT = 5;
81 const int MARGIN = 2;
82 const int MININDENT = 10;
83 const int BTNWIDTH = 11;
84 const int BTNHEIGHT = 11;
85
86 const wxChar* wxTreeListCtrlNameStr = wxT("treelistctrl");
87
88 static wxTreeListColumnInfo wxInvalidTreeListColumnInfo;
89
90
91 // ---------------------------------------------------------------------------
92 // private classes
93 // ---------------------------------------------------------------------------
94 //-----------------------------------------------------------------------------
95 // wxTreeListHeaderWindow (internal)
96 //-----------------------------------------------------------------------------
97
98 class wxTreeListHeaderWindow : public wxWindow
99 {
100 protected:
101 wxTreeListMainWindow *m_owner;
102 wxCursor *m_currentCursor;
103 wxCursor *m_resizeCursor;
104 bool m_isDragging;
105
106 // column being resized
107 int m_column;
108
109 // divider line position in logical (unscrolled) coords
110 int m_currentX;
111
112 // minimal position beyond which the divider line can't be dragged in
113 // logical coords
114 int m_minX;
115
116 wxArrayTreeListColumnInfo m_columns;
117
118 // total width of the columns
119 int m_total_col_width;
120
121
122 public:
123 wxTreeListHeaderWindow();
124
125 wxTreeListHeaderWindow( wxWindow *win,
126 wxWindowID id,
127 wxTreeListMainWindow *owner,
128 const wxPoint &pos = wxDefaultPosition,
129 const wxSize &size = wxDefaultSize,
130 long style = 0,
131 const wxString &name = wxT("wxtreelistctrlcolumntitles") );
132
133 virtual ~wxTreeListHeaderWindow();
134
135 void DoDrawRect( wxDC *dc, int x, int y, int w, int h );
136 void DrawCurrent();
137 void AdjustDC(wxDC& dc);
138
139 void OnPaint( wxPaintEvent &event );
140 void OnMouse( wxMouseEvent &event );
141 void OnSetFocus( wxFocusEvent &event );
142
143
144 // columns manipulation
145
146 size_t GetColumnCount() const { return m_columns.GetCount(); }
147
148 void AddColumn(const wxTreeListColumnInfo& col);
149
150 void InsertColumn(size_t before, const wxTreeListColumnInfo& col);
151
152 void RemoveColumn(size_t column);
153
154 void SetColumn(size_t column, const wxTreeListColumnInfo& info);
155 const wxTreeListColumnInfo& GetColumn(size_t column) const
156 {
157 wxCHECK_MSG(column < GetColumnCount(), wxInvalidTreeListColumnInfo, wxT("Invalid column"));
158 return m_columns[column];
159 }
160 wxTreeListColumnInfo& GetColumn(size_t column)
161 {
162 wxCHECK_MSG(column < GetColumnCount(), wxInvalidTreeListColumnInfo, wxT("Invalid column"));
163 return m_columns[column];
164 }
165
166 void SetColumnWidth(size_t column, size_t width);
167
168 void SetColumnText(size_t column, const wxString& text)
169 {
170 wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
171 m_columns[column].SetText(text);
172 }
173
174 void SetColumnShown(size_t column, bool shown)
175 {
176 wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
177 m_columns[column].SetShown(shown);
178 }
179
180 wxString GetColumnText(size_t column) const
181 {
182 wxCHECK_MSG(column < GetColumnCount(), wxEmptyString, wxT("Invalid column"));
183 return m_columns[column].GetText();
184 }
185
186 int GetColumnWidth(size_t column) const
187 {
188 wxCHECK_MSG(column < GetColumnCount(), -1, wxT("Invalid column"));
189 return m_columns[column].GetWidth();
190 }
191
192 int GetWidth() const { return m_total_col_width; }
193
194 int GetColumnShown(size_t column) const
195 {
196 wxCHECK_MSG(column < GetColumnCount(), -1, wxT("Invalid column"));
197 return m_columns[column].GetShown();
198 }
199
200 // needs refresh
201 bool m_dirty;
202
203 private:
204 // common part of all ctors
205 void Init();
206
207 void SendListEvent(wxEventType type, wxPoint pos);
208
209 DECLARE_DYNAMIC_CLASS(wxTreeListHeaderWindow)
210 DECLARE_EVENT_TABLE()
211 };
212
213
214 // this is the "true" control
215 class wxTreeListMainWindow: public wxScrolledWindow
216 {
217 public:
218 // creation
219 // --------
220 wxTreeListMainWindow() { Init(); }
221
222 wxTreeListMainWindow(wxTreeListCtrl *parent, wxWindowID id = -1,
223 const wxPoint& pos = wxDefaultPosition,
224 const wxSize& size = wxDefaultSize,
225 long style = wxTR_DEFAULT_STYLE,
226 const wxValidator &validator = wxDefaultValidator,
227 const wxString& name = wxT("wxtreelistmainwindow"))
228 {
229 Init();
230 Create(parent, id, pos, size, style, validator, name);
231 }
232
233 virtual ~wxTreeListMainWindow();
234
235 bool Create(wxTreeListCtrl *parent, wxWindowID id = -1,
236 const wxPoint& pos = wxDefaultPosition,
237 const wxSize& size = wxDefaultSize,
238 long style = wxTR_DEFAULT_STYLE,
239 const wxValidator &validator = wxDefaultValidator,
240 const wxString& name = wxT("wxtreelistctrl"));
241
242 // accessors
243 // ---------
244
245 // get the total number of items in the control
246 size_t GetCount() const;
247
248 // indent is the number of pixels the children are indented relative to
249 // the parents position. SetIndent() also redraws the control
250 // immediately.
251 unsigned int GetIndent() const { return m_indent; }
252 void SetIndent(unsigned int indent);
253
254 // see wxTreeListCtrl for the meaning
255 unsigned int GetLineSpacing() const { return m_linespacing; }
256 void SetLineSpacing(unsigned int spacing);
257
258 // image list: these functions allow to associate an image list with
259 // the control and retrieve it. Note that when assigned with
260 // SetImageList, the control does _not_ delete
261 // the associated image list when it's deleted in order to allow image
262 // lists to be shared between different controls. If you use
263 // AssignImageList, the control _does_ delete the image list.
264
265 // The normal image list is for the icons which correspond to the
266 // normal tree item state (whether it is selected or not).
267 // Additionally, the application might choose to show a state icon
268 // which corresponds to an app-defined item state (for example,
269 // checked/unchecked) which are taken from the state image list.
270 wxImageList *GetImageList() const;
271 wxImageList *GetStateImageList() const;
272 wxImageList *GetButtonsImageList() const;
273
274 void SetImageList(wxImageList *imageList);
275 void SetStateImageList(wxImageList *imageList);
276 void SetButtonsImageList(wxImageList *imageList);
277 void AssignImageList(wxImageList *imageList);
278 void AssignStateImageList(wxImageList *imageList);
279 void AssignButtonsImageList(wxImageList *imageList);
280
281 // Functions to work with tree ctrl items.
282
283 // accessors
284 // ---------
285
286 // retrieve item's label
287 wxString GetItemText(const wxTreeItemId& item) const
288 { return GetItemText(item, GetMainColumn()); }
289 // get one of the images associated with the item (normal by default)
290 int GetItemImage(const wxTreeItemId& item,
291 wxTreeItemIcon which = wxTreeItemIcon_Normal) const
292 { return GetItemImage(item, GetMainColumn(), which); }
293
294 // get the data associated with the item
295 wxTreeItemData *GetItemData(const wxTreeItemId& item) const;
296
297 bool GetItemBold(const wxTreeItemId& item) const;
298 wxColour GetItemTextColour(const wxTreeItemId& item) const;
299 wxColour GetItemBackgroundColour(const wxTreeItemId& item) const;
300 wxFont GetItemFont(const wxTreeItemId& item) const;
301
302 // modifiers
303 // ---------
304
305 // set item's label
306 void SetItemText(const wxTreeItemId& item, const wxString& text)
307 { SetItemText(item, GetMainColumn(), text); }
308
309 // get one of the images associated with the item (normal by default)
310 void SetItemImage(const wxTreeItemId& item, int image,
311 wxTreeItemIcon which = wxTreeItemIcon_Normal)
312 { SetItemImage(item, GetMainColumn(), image, which); }
313
314 // associate some data with the item
315 void SetItemData(const wxTreeItemId& item, wxTreeItemData *data);
316
317 // force appearance of [+] button near the item. This is useful to
318 // allow the user to expand the items which don't have any children now
319 // - but instead add them only when needed, thus minimizing memory
320 // usage and loading time.
321 void SetItemHasChildren(const wxTreeItemId& item, bool has = TRUE);
322
323 // the item will be shown in bold
324 void SetItemBold(const wxTreeItemId& item, bool bold = TRUE);
325
326 // set the item's text colour
327 void SetItemTextColour(const wxTreeItemId& item, const wxColour& colour);
328
329 // set the item's background colour
330 void SetItemBackgroundColour(const wxTreeItemId& item, const wxColour& colour);
331
332 // set the item's font (should be of the same height for all items)
333 void SetItemFont(const wxTreeItemId& item, const wxFont& font);
334
335 // set the window font
336 virtual bool SetFont( const wxFont &font );
337
338 // set the styles. No need to specify a GetWindowStyle here since
339 // the base wxWindow member function will do it for us
340 void SetWindowStyle(const long styles);
341
342 // item status inquiries
343 // ---------------------
344
345 // is the item visible (it might be outside the view or not expanded)?
346 bool IsVisible(const wxTreeItemId& item) const;
347 // does the item has any children?
348 bool HasChildren(const wxTreeItemId& item) const
349 { return ItemHasChildren(item); }
350 bool ItemHasChildren(const wxTreeItemId& item) const;
351 // is the item expanded (only makes sense if HasChildren())?
352 bool IsExpanded(const wxTreeItemId& item) const;
353 // is this item currently selected (the same as has focus)?
354 bool IsSelected(const wxTreeItemId& item) const;
355 // is item text in bold font?
356 bool IsBold(const wxTreeItemId& item) const;
357 // does the layout include space for a button?
358
359 // number of children
360 // ------------------
361
362 // if 'recursively' is FALSE, only immediate children count, otherwise
363 // the returned number is the number of all items in this branch
364 size_t GetChildrenCount(const wxTreeItemId& item, bool recursively = TRUE);
365
366 // navigation
367 // ----------
368
369 // wxTreeItemId.IsOk() will return FALSE if there is no such item
370
371 // get the root tree item
372 wxTreeItemId GetRootItem() const { return m_anchor; }
373
374 // get the item currently selected (may return NULL if no selection)
375 wxTreeItemId GetSelection() const { return m_current; }
376
377 // get the items currently selected, return the number of such item
378 size_t GetSelections(wxArrayTreeItemIds&) const;
379
380 // get the parent of this item (may return NULL if root)
381 wxTreeItemId GetItemParent(const wxTreeItemId& item) const;
382
383 // for this enumeration function you must pass in a "cookie" parameter
384 // which is opaque for the application but is necessary for the library
385 // to make these functions reentrant (i.e. allow more than one
386 // enumeration on one and the same object simultaneously). Of course,
387 // the "cookie" passed to GetFirstChild() and GetNextChild() should be
388 // the same!
389
390 // get the first child of this item
391 #if !wxCHECK_VERSION(2, 5, 0)
392 wxTreeItemId GetFirstChild(const wxTreeItemId& item, long& cookie) const;
393 #else
394 wxTreeItemId GetFirstChild(const wxTreeItemId& item, wxTreeItemIdValue& cookie) const;
395 #endif
396 // get the next child
397 #if !wxCHECK_VERSION(2, 5, 0)
398 wxTreeItemId GetNextChild(const wxTreeItemId& item, long& cookie) const;
399 #else
400 wxTreeItemId GetNextChild(const wxTreeItemId& item, wxTreeItemIdValue& cookie) const;
401 #endif
402 // get the prev child
403 #if !wxCHECK_VERSION(2, 5, 0)
404 wxTreeItemId GetPrevChild(const wxTreeItemId& item, long& cookie) const;
405 #else
406 wxTreeItemId GetPrevChild(const wxTreeItemId& item, wxTreeItemIdValue& cookie) const;
407 #endif
408 // get the last child of this item - this method doesn't use cookies
409 wxTreeItemId GetLastChild(const wxTreeItemId& item) const;
410
411 // get the next sibling of this item
412 wxTreeItemId GetNextSibling(const wxTreeItemId& item) const;
413 // get the previous sibling
414 wxTreeItemId GetPrevSibling(const wxTreeItemId& item) const;
415
416 // get first visible item
417 wxTreeItemId GetFirstVisibleItem() const;
418 // get the next visible item: item must be visible itself!
419 // see IsVisible() and wxTreeCtrl::GetFirstVisibleItem()
420 wxTreeItemId GetNextVisible(const wxTreeItemId& item) const;
421 // get the previous visible item: item must be visible itself!
422 wxTreeItemId GetPrevVisible(const wxTreeItemId& item) const;
423
424 // Only for internal use right now, but should probably be public
425 wxTreeItemId GetNext(const wxTreeItemId& item) const;
426
427 // operations
428 // ----------
429
430 // add the root node to the tree
431 wxTreeItemId AddRoot(const wxString& text,
432 int image = -1, int selectedImage = -1,
433 wxTreeItemData *data = NULL);
434
435 // insert a new item in as the first child of the parent
436 wxTreeItemId PrependItem(const wxTreeItemId& parent,
437 const wxString& text,
438 int image = -1, int selectedImage = -1,
439 wxTreeItemData *data = NULL);
440
441 // insert a new item after a given one
442 wxTreeItemId InsertItem(const wxTreeItemId& parent,
443 const wxTreeItemId& idPrevious,
444 const wxString& text,
445 int image = -1, int selectedImage = -1,
446 wxTreeItemData *data = NULL);
447
448 // insert a new item before the one with the given index
449 wxTreeItemId InsertItem(const wxTreeItemId& parent,
450 size_t index,
451 const wxString& text,
452 int image = -1, int selectedImage = -1,
453 wxTreeItemData *data = NULL);
454
455 // insert a new item in as the last child of the parent
456 wxTreeItemId AppendItem(const wxTreeItemId& parent,
457 const wxString& text,
458 int image = -1, int selectedImage = -1,
459 wxTreeItemData *data = NULL);
460
461 // delete this item and associated data if any
462 void Delete(const wxTreeItemId& item);
463 // delete all children (but don't delete the item itself)
464 // NB: this won't send wxEVT_COMMAND_TREE_ITEM_DELETED events
465 void DeleteChildren(const wxTreeItemId& item);
466 // delete all items from the tree
467 // NB: this won't send wxEVT_COMMAND_TREE_ITEM_DELETED events
468 void DeleteAllItems();
469
470 // expand this item
471 void Expand(const wxTreeItemId& item);
472 // expand this item and all subitems recursively
473 void ExpandAll(const wxTreeItemId& item);
474 // collapse the item without removing its children
475 void Collapse(const wxTreeItemId& item);
476 // collapse the item and remove all children
477 void CollapseAndReset(const wxTreeItemId& item);
478 // toggles the current state
479 void Toggle(const wxTreeItemId& item);
480
481 // remove the selection from currently selected item (if any)
482 void Unselect();
483 void UnselectAll();
484 // select this item
485 void SelectItem(const wxTreeItemId& item, bool unselect_others=TRUE,
486 bool extended_select=FALSE);
487 void SelectAll(bool extended_select=FALSE);
488 // make sure this item is visible (expanding the parent item and/or
489 // scrolling to this item if necessary)
490 void EnsureVisible(const wxTreeItemId& item);
491 // scroll to this item (but don't expand its parent)
492 void ScrollTo(const wxTreeItemId& item);
493 void AdjustMyScrollbars();
494
495 // The first function is more portable (because easier to implement
496 // on other platforms), but the second one returns some extra info.
497 wxTreeItemId HitTest(const wxPoint& point)
498 { int dummy; return HitTest(point, dummy); }
499 wxTreeItemId HitTest(const wxPoint& point, int& flags)
500 { int col; return HitTest(point, flags, col); }
501 // ALB
502 wxTreeItemId HitTest(const wxPoint& point, int& flags, int& column);
503
504
505 // get the bounding rectangle of the item (or of its label only)
506 bool GetBoundingRect(const wxTreeItemId& item,
507 wxRect& rect,
508 bool textOnly = FALSE) const;
509
510 // Start editing the item label: this (temporarily) replaces the item
511 // with a one line edit control. The item will be selected if it hadn't
512 // been before.
513 void EditLabel( const wxTreeItemId& item ) { Edit( item ); }
514 void Edit( const wxTreeItemId& item );
515
516 // sorting
517 // this function is called to compare 2 items and should return -1, 0
518 // or +1 if the first item is less than, equal to or greater than the
519 // second one. The base class version performs alphabetic comparaison
520 // of item labels (GetText)
521 virtual int OnCompareItems(const wxTreeItemId& item1,
522 const wxTreeItemId& item2);
523 // sort the children of this item using OnCompareItems
524 //
525 // NB: this function is not reentrant and not MT-safe (FIXME)!
526 void SortChildren(const wxTreeItemId& item);
527
528 // searching
529 wxTreeItemId FindItem (const wxTreeItemId& item, const wxString& str, int flags = 0);
530
531 // deprecated functions: use Set/GetItemImage directly
532 // get the selected item image
533 int GetItemSelectedImage(const wxTreeItemId& item) const
534 { return GetItemImage(item, wxTreeItemIcon_Selected); }
535 // set the selected item image
536 void SetItemSelectedImage(const wxTreeItemId& item, int image)
537 { SetItemImage(item, image, wxTreeItemIcon_Selected); }
538
539 // implementation only from now on
540
541 // overridden base class virtuals
542 virtual bool SetBackgroundColour(const wxColour& colour);
543 virtual bool SetForegroundColour(const wxColour& colour);
544
545 // callbacks
546 void OnPaint( wxPaintEvent &event );
547 void OnSetFocus( wxFocusEvent &event );
548 void OnKillFocus( wxFocusEvent &event );
549 void OnChar( wxKeyEvent &event );
550 void OnMouse( wxMouseEvent &event );
551 void OnIdle( wxIdleEvent &event );
552 void OnSize(wxSizeEvent& event); // ALB
553 void OnScroll(wxScrollWinEvent& event); // ALB
554
555 // implementation helpers
556 void SendDeleteEvent(wxTreeListItem *itemBeingDeleted);
557
558 void DrawBorder(const wxTreeItemId& item);
559 void DrawLine(const wxTreeItemId& item, bool below);
560
561 size_t GetColumnCount() const
562 { return m_owner->GetHeaderWindow()->GetColumnCount(); }
563
564 void SetMainColumn(size_t column)
565 {
566 if(column < GetColumnCount())
567 m_main_column = column;
568 }
569 size_t GetMainColumn() const { return m_main_column; }
570
571 void SetItemText(const wxTreeItemId& item, size_t column,
572 const wxString& text);
573 wxString GetItemText(const wxTreeItemId& item, size_t column) const;
574
575 void SetItemImage(const wxTreeItemId& item, size_t column, int image,
576 wxTreeItemIcon which = wxTreeItemIcon_Normal);
577 int GetItemImage(const wxTreeItemId& item, size_t column,
578 wxTreeItemIcon which = wxTreeItemIcon_Normal) const;
579
580 void SetFocus();
581
582 protected:
583 wxTreeListCtrl* m_owner; // ALB
584
585 size_t m_main_column; // ALB
586
587 friend class wxTreeListItem;
588 friend class wxTreeListRenameTimer;
589 friend class wxTreeListTextCtrl;
590
591 wxFont m_normalFont;
592 wxFont m_boldFont;
593
594 wxTreeListItem *m_anchor;
595 wxTreeListItem *m_current, *m_key_current, *m_currentEdit;
596 int m_btnWidth, m_btnWidth2;
597 int m_btnHeight, m_btnHeight2;
598 int m_imgWidth, m_imgWidth2;
599 int m_imgHeight, m_imgHeight2;
600 unsigned short m_indent;
601 int m_lineHeight;
602 unsigned short m_linespacing;
603 wxPen m_dottedPen;
604 wxBrush *m_hilightBrush,
605 *m_hilightUnfocusedBrush;
606 bool m_hasFocus;
607 public:
608 bool m_dirty;
609 protected:
610 bool m_ownsImageListNormal,
611 m_ownsImageListState,
612 m_ownsImageListButtons;
613 bool m_isDragging; // true between BEGIN/END drag events
614 bool m_renameAccept;
615 bool m_lastOnSame; // last click on the same item as prev
616 wxImageList *m_imageListNormal,
617 *m_imageListState,
618 *m_imageListButtons;
619
620 int m_dragCount;
621 wxPoint m_dragStart;
622 wxTreeListItem *m_dropTarget;
623 wxCursor m_oldCursor; // cursor is changed while dragging
624 wxTreeListItem *m_oldSelection;
625
626 wxTimer *m_renameTimer;
627 wxString m_renameRes;
628
629 // char navigation
630 wxTimer *m_findTimer;
631 wxString m_findStr;
632
633 // the common part of all ctors
634 void Init();
635
636 // misc helpers
637 wxTreeItemId DoInsertItem(const wxTreeItemId& parent,
638 size_t previous,
639 const wxString& text,
640 int image, int selectedImage,
641 wxTreeItemData *data);
642 bool HasButtons(void) const
643 { return (m_imageListButtons != NULL) ||
644 HasFlag (wxTR_TWIST_BUTTONS|wxTR_HAS_BUTTONS); }
645
646 protected:
647 void CalculateLineHeight();
648 int GetLineHeight(wxTreeListItem *item) const;
649 void PaintLevel( wxTreeListItem *item, wxDC& dc, int level, int &y,
650 int x_colstart);
651 void PaintItem( wxTreeListItem *item, wxDC& dc);
652
653 void CalculateLevel( wxTreeListItem *item, wxDC &dc, int level, int &y,
654 int x_colstart);
655 void CalculatePositions();
656 void CalculateSize( wxTreeListItem *item, wxDC &dc );
657
658 void RefreshSubtree( wxTreeListItem *item );
659 void RefreshLine( wxTreeListItem *item );
660
661 // redraw all selected items
662 void RefreshSelected();
663
664 // RefreshSelected() recursive helper
665 void RefreshSelectedUnder(wxTreeListItem *item);
666
667 void OnRenameTimer();
668 void OnRenameAccept();
669
670 void FillArray(wxTreeListItem*, wxArrayTreeItemIds&) const;
671 void SelectItemRange( wxTreeListItem *item1, wxTreeListItem *item2 );
672 bool TagAllChildrenUntilLast(wxTreeListItem *crt_item,
673 wxTreeListItem *last_item, bool select);
674 bool TagNextChildren(wxTreeListItem *crt_item, wxTreeListItem *last_item,
675 bool select);
676 void UnselectAllChildren( wxTreeListItem *item );
677
678 void DrawDropEffect(wxTreeListItem *item);
679
680 private:
681 DECLARE_EVENT_TABLE()
682 DECLARE_DYNAMIC_CLASS(wxTreeListMainWindow)
683 };
684
685
686 // timer used for enabling in-place edit
687 class wxTreeListRenameTimer: public wxTimer
688 {
689 public:
690 wxTreeListRenameTimer( wxTreeListMainWindow *owner );
691
692 void Notify();
693
694 private:
695 wxTreeListMainWindow *m_owner;
696 };
697
698 // control used for in-place edit
699 class wxTreeListTextCtrl: public wxTextCtrl
700 {
701 public:
702 wxTreeListTextCtrl( wxWindow *parent,
703 const wxWindowID id,
704 bool *accept,
705 wxString *res,
706 wxTreeListMainWindow *owner,
707 const wxString &value = wxEmptyString,
708 const wxPoint &pos = wxDefaultPosition,
709 const wxSize &size = wxDefaultSize,
710 int style = wxSIMPLE_BORDER,
711 const wxValidator& validator = wxDefaultValidator,
712 const wxString &name = wxTextCtrlNameStr );
713
714 void OnChar( wxKeyEvent &event );
715 void OnKeyUp( wxKeyEvent &event );
716 void OnKillFocus( wxFocusEvent &event );
717
718 private:
719 bool *m_accept;
720 wxString *m_res;
721 wxTreeListMainWindow *m_owner;
722 wxString m_startValue;
723 bool m_finished;
724
725 DECLARE_EVENT_TABLE()
726 };
727
728 // a tree item
729 class wxTreeListItem
730 {
731 public:
732 // ctors & dtor
733 wxTreeListItem() { m_data = NULL; }
734 wxTreeListItem( wxTreeListMainWindow *owner,
735 wxTreeListItem *parent,
736 const wxArrayString& text,
737 int image,
738 int selImage,
739 wxTreeItemData *data );
740
741 ~wxTreeListItem();
742
743 // trivial accessors
744 wxArrayTreeListItems& GetChildren() { return m_children; }
745
746 const wxString GetText() const
747 {
748 if(m_text.GetCount() > 0) return m_text[0];
749 return wxEmptyString;
750 }
751 const wxString GetText(size_t col) const
752 {
753 if(m_text.GetCount() > col) return m_text[col];
754 return wxEmptyString;
755 }
756 int GetImage(wxTreeItemIcon which = wxTreeItemIcon_Normal) const
757 { return m_images[which]; }
758 int GetImage(size_t col, wxTreeItemIcon which=wxTreeItemIcon_Normal) const
759 {
760 if(col == m_owner->GetMainColumn()) return m_images[which];
761 if(col < m_col_images.GetCount()) return m_col_images[col];
762 return NO_IMAGE;
763 }
764 wxTreeItemData *GetData() const { return m_data; }
765
766 // returns the current image for the item (depending on its
767 // selected/expanded/whatever state)
768 int GetCurrentImage() const;
769
770 void SetText( const wxString &text );
771 void SetText(size_t col, const wxString& text) // ALB
772 {
773 if(col < m_text.GetCount())
774 m_text[col] = text;
775 else if(col < m_owner->GetColumnCount()) {
776 int howmany = m_owner->GetColumnCount();
777 for(int i = m_text.GetCount(); i < howmany; ++i)
778 m_text.Add(wxEmptyString);
779 m_text[col] = text;
780 }
781 }
782 void SetImage(int image, wxTreeItemIcon which) { m_images[which] = image; }
783 void SetImage(size_t col, int image, wxTreeItemIcon which)
784 {
785 if(col == m_owner->GetMainColumn()) m_images[which] = image;
786 else if(col < m_col_images.GetCount())
787 m_col_images[col] = image;
788 else if(col < m_owner->GetColumnCount()) {
789 int howmany = m_owner->GetColumnCount();
790 for(int i = m_col_images.GetCount(); i < howmany; ++i)
791 m_col_images.Add(NO_IMAGE);
792 m_col_images[col] = image;
793 }
794 }
795
796 void SetData(wxTreeItemData *data) { m_data = data; }
797
798 void SetHasPlus(bool has = TRUE) { m_hasPlus = has; }
799
800 void SetBold(bool bold) { m_isBold = bold; }
801
802 int GetX() const { return m_x; }
803 int GetY() const { return m_y; }
804
805 void SetX(int x) { m_x = x; }
806 void SetY(int y) { m_y = y; }
807
808 int GetHeight() const { return m_height; }
809 int GetWidth() const { return m_width; }
810
811 void SetHeight(int h) { m_height = h; }
812 void SetWidth(int w) { m_width = w; }
813
814 wxTreeListItem *GetItemParent() const { return m_parent; }
815
816 // operations
817 // deletes all children notifying the treectrl about it if !NULL
818 // pointer given
819 void DeleteChildren(wxTreeListMainWindow *tree = NULL);
820
821 // get count of all children (and grand children if 'recursively')
822 size_t GetChildrenCount(bool recursively = TRUE) const;
823
824 void Insert(wxTreeListItem *child, size_t index)
825 { m_children.Insert(child, index); }
826
827 void GetSize( int &x, int &y, const wxTreeListMainWindow* );
828
829 // return the item at given position (or NULL if no item), onButton is
830 // TRUE if the point belongs to the item's button, otherwise it lies
831 // on the button's label
832 wxTreeListItem *HitTest( const wxPoint& point,
833 const wxTreeListMainWindow *,
834 int &flags,
835 int level );
836 wxTreeListItem *HitTest( const wxPoint& point,
837 const wxTreeListMainWindow *,
838 int &flags, int& column /*ALB*/,
839 int level );
840
841 void Expand() { m_isCollapsed = FALSE; }
842 void Collapse() { m_isCollapsed = TRUE; }
843
844 void SetHilight( bool set = TRUE ) { m_hasHilight = set; }
845
846 // status inquiries
847 bool HasChildren() const { return !m_children.IsEmpty(); }
848 bool IsSelected() const { return m_hasHilight != 0; }
849 bool IsExpanded() const { return !m_isCollapsed; }
850 bool HasPlus() const { return m_hasPlus || HasChildren(); }
851 bool IsBold() const { return m_isBold != 0; }
852
853 // attributes
854 // get them - may be NULL
855 wxTreeItemAttr *GetAttributes() const { return m_attr; }
856 // get them ensuring that the pointer is not NULL
857 wxTreeItemAttr& Attr()
858 {
859 if ( !m_attr )
860 {
861 m_attr = new wxTreeItemAttr;
862 m_ownsAttr = TRUE;
863 }
864 return *m_attr;
865 }
866 // set them
867 void SetAttributes(wxTreeItemAttr *attr)
868 {
869 if ( m_ownsAttr ) delete m_attr;
870 m_attr = attr;
871 m_ownsAttr = FALSE;
872 }
873 // set them and delete when done
874 void AssignAttributes(wxTreeItemAttr *attr)
875 {
876 SetAttributes(attr);
877 m_ownsAttr = TRUE;
878 }
879
880 private:
881 wxTreeListMainWindow *m_owner; // control the item belongs to
882
883 // since there can be very many of these, we save size by chosing
884 // the smallest representation for the elements and by ordering
885 // the members to avoid padding.
886 wxArrayString m_text; // labels to be rendered for item
887
888 wxTreeItemData *m_data; // user-provided data
889
890 wxArrayTreeListItems m_children; // list of children
891 wxTreeListItem *m_parent; // parent of this item
892
893 wxTreeItemAttr *m_attr; // attributes???
894
895 // tree ctrl images for the normal, selected, expanded and
896 // expanded+selected states
897 short m_images[wxTreeItemIcon_Max];
898 wxArrayShort m_col_images; // images for the various columns (!= main)
899
900 wxCoord m_x; // (virtual) offset from top
901 wxCoord m_y; // (virtual) offset from left
902 short m_width; // width of this item
903 unsigned char m_height; // height of this item
904
905 // use bitfields to save size
906 int m_isCollapsed :1;
907 int m_hasHilight :1; // same as focused
908 int m_hasPlus :1; // used for item which doesn't have
909 // children but has a [+] button
910 int m_isBold :1; // render the label in bold font
911 int m_ownsAttr :1; // delete attribute when done
912 };
913
914 // ===========================================================================
915 // implementation
916 // ===========================================================================
917
918 // ----------------------------------------------------------------------------
919 // private functions
920 // ----------------------------------------------------------------------------
921
922 // translate the key or mouse event flags to the type of selection we're
923 // dealing with
924 static void EventFlagsToSelType(long style,
925 bool shiftDown,
926 bool ctrlDown,
927 bool &is_multiple,
928 bool &extended_select,
929 bool &unselect_others)
930 {
931 is_multiple = (style & wxTR_MULTIPLE) != 0;
932 extended_select = shiftDown && is_multiple;
933 unselect_others = !(extended_select || (ctrlDown && is_multiple));
934 }
935
936 // ---------------------------------------------------------------------------
937 // wxTreeListRenameTimer (internal)
938 // ---------------------------------------------------------------------------
939
940 wxTreeListRenameTimer::wxTreeListRenameTimer( wxTreeListMainWindow *owner )
941 {
942 m_owner = owner;
943 }
944
945 void wxTreeListRenameTimer::Notify()
946 {
947 m_owner->OnRenameTimer();
948 }
949
950 //-----------------------------------------------------------------------------
951 // wxTreeListTextCtrl (internal)
952 //-----------------------------------------------------------------------------
953
954 BEGIN_EVENT_TABLE(wxTreeListTextCtrl,wxTextCtrl)
955 EVT_CHAR (wxTreeListTextCtrl::OnChar)
956 EVT_KEY_UP (wxTreeListTextCtrl::OnKeyUp)
957 EVT_KILL_FOCUS (wxTreeListTextCtrl::OnKillFocus)
958 END_EVENT_TABLE()
959
960 wxTreeListTextCtrl::wxTreeListTextCtrl( wxWindow *parent,
961 const wxWindowID id,
962 bool *accept,
963 wxString *res,
964 wxTreeListMainWindow *owner,
965 const wxString &value,
966 const wxPoint &pos,
967 const wxSize &size,
968 int style,
969 const wxValidator& validator,
970 const wxString &name )
971 : wxTextCtrl( parent, id, value, pos, size, style, validator, name )
972 {
973 m_res = res;
974 m_accept = accept;
975 m_owner = owner;
976 (*m_accept) = FALSE;
977 (*m_res) = wxEmptyString;
978 m_startValue = value;
979 m_finished = FALSE;
980 }
981
982 void wxTreeListTextCtrl::OnChar( wxKeyEvent &event )
983 {
984 if (event.m_keyCode == WXK_RETURN)
985 {
986 (*m_accept) = TRUE;
987 (*m_res) = GetValue();
988
989 if ((*m_res) != m_startValue)
990 m_owner->OnRenameAccept();
991
992 if (!wxPendingDelete.Member(this))
993 wxPendingDelete.Append(this);
994
995 m_finished = TRUE;
996 m_owner->SetFocus(); // This doesn't work. TODO.
997
998 return;
999 }
1000 if (event.m_keyCode == WXK_ESCAPE)
1001 {
1002 (*m_accept) = FALSE;
1003 (*m_res) = wxEmptyString;
1004
1005 if (!wxPendingDelete.Member(this))
1006 wxPendingDelete.Append(this);
1007
1008 m_finished = TRUE;
1009 m_owner->SetFocus(); // This doesn't work. TODO.
1010
1011 return;
1012 }
1013 event.Skip();
1014 }
1015
1016 void wxTreeListTextCtrl::OnKeyUp( wxKeyEvent &event )
1017 {
1018 if (m_finished)
1019 {
1020 event.Skip();
1021 return;
1022 }
1023
1024 // auto-grow the textctrl:
1025 wxSize parentSize = m_owner->GetSize();
1026 wxPoint myPos = GetPosition();
1027 wxSize mySize = GetSize();
1028 int sx, sy;
1029 GetTextExtent(GetValue() + _T("M"), &sx, &sy);
1030 if (myPos.x + sx > parentSize.x) sx = parentSize.x - myPos.x;
1031 if (mySize.x > sx) sx = mySize.x;
1032 SetSize(sx, -1);
1033
1034 event.Skip();
1035 }
1036
1037 void wxTreeListTextCtrl::OnKillFocus( wxFocusEvent &event )
1038 {
1039 if (m_finished)
1040 {
1041 event.Skip();
1042 return;
1043 }
1044
1045 if (!wxPendingDelete.Member(this))
1046 wxPendingDelete.Append(this);
1047
1048 (*m_accept) = TRUE;
1049 (*m_res) = GetValue();
1050
1051 if ((*m_res) != m_startValue)
1052 m_owner->OnRenameAccept();
1053 }
1054
1055 //-----------------------------------------------------------------------------
1056 // wxTreeListHeaderWindow
1057 //-----------------------------------------------------------------------------
1058
1059 IMPLEMENT_DYNAMIC_CLASS(wxTreeListHeaderWindow,wxWindow);
1060
1061 BEGIN_EVENT_TABLE(wxTreeListHeaderWindow,wxWindow)
1062 EVT_PAINT (wxTreeListHeaderWindow::OnPaint)
1063 EVT_MOUSE_EVENTS (wxTreeListHeaderWindow::OnMouse)
1064 EVT_SET_FOCUS (wxTreeListHeaderWindow::OnSetFocus)
1065 END_EVENT_TABLE()
1066
1067 void wxTreeListHeaderWindow::Init()
1068 {
1069 m_currentCursor = (wxCursor *) NULL;
1070 m_isDragging = FALSE;
1071 m_dirty = FALSE;
1072 m_total_col_width = 0;
1073 }
1074
1075 wxTreeListHeaderWindow::wxTreeListHeaderWindow()
1076 {
1077 Init();
1078
1079 m_owner = (wxTreeListMainWindow *) NULL;
1080 m_resizeCursor = (wxCursor *) NULL;
1081 }
1082
1083 wxTreeListHeaderWindow::wxTreeListHeaderWindow( wxWindow *win,
1084 wxWindowID id,
1085 wxTreeListMainWindow *owner,
1086 const wxPoint& pos,
1087 const wxSize& size,
1088 long style,
1089 const wxString &name )
1090 : wxWindow( win, id, pos, size, style, name )
1091 {
1092 Init();
1093
1094 m_owner = owner;
1095 m_resizeCursor = new wxCursor(wxCURSOR_SIZEWE);
1096
1097 SetBackgroundColour(wxSystemSettings::GetSystemColour(
1098 wxSYS_COLOUR_BTNFACE));
1099 }
1100
1101 wxTreeListHeaderWindow::~wxTreeListHeaderWindow()
1102 {
1103 delete m_resizeCursor;
1104 }
1105
1106 void wxTreeListHeaderWindow::DoDrawRect( wxDC *dc, int x, int y, int w, int h )
1107 {
1108 #ifdef __WXGTK__
1109 GtkStateType state = m_parent->IsEnabled() ? GTK_STATE_NORMAL
1110 : GTK_STATE_INSENSITIVE;
1111
1112 x = dc->XLOG2DEV( x );
1113
1114 gtk_paint_box (m_wxwindow->style, GTK_PIZZA(m_wxwindow)->bin_window,
1115 state, GTK_SHADOW_OUT,
1116 (GdkRectangle*) NULL, m_wxwindow, "button",
1117 x-1, y-1, w+2, h+2);
1118 #elif defined( __WXMAC__ )
1119 const int m_corner = 1;
1120
1121 dc->SetBrush( *wxTRANSPARENT_BRUSH );
1122
1123 dc->SetPen( wxPen(wxSystemSettings::GetSystemColour(
1124 wxSYS_COLOUR_BTNSHADOW), 1, wxSOLID));
1125 dc->DrawLine( x+w-m_corner+1, y, x+w, y+h ); // right (outer)
1126 dc->DrawRectangle( x, y+h, w+1, 1 ); // bottom (outer)
1127
1128 wxPen pen( wxColour( 0x88 , 0x88 , 0x88 ), 1, wxSOLID );
1129
1130 dc->SetPen( pen );
1131 dc->DrawLine( x+w-m_corner, y, x+w-1, y+h ); // right (inner)
1132 dc->DrawRectangle( x+1, y+h-1, w-2, 1 ); // bottom (inner)
1133
1134 dc->SetPen( *wxWHITE_PEN );
1135 dc->DrawRectangle( x, y, w-m_corner+1, 1 ); // top (outer)
1136 dc->DrawRectangle( x, y, 1, h ); // left (outer)
1137 dc->DrawLine( x, y+h-1, x+1, y+h-1 );
1138 dc->DrawLine( x+w-1, y, x+w-1, y+1 );
1139 #else // !GTK, !Mac
1140 const int m_corner = 1;
1141
1142 dc->SetBrush( *wxTRANSPARENT_BRUSH );
1143
1144 dc->SetPen( *wxBLACK_PEN );
1145 dc->DrawLine( x+w-m_corner+1, y, x+w, y+h ); // right (outer)
1146 dc->DrawRectangle( x, y+h, w+1, 1 ); // bottom (outer)
1147
1148 wxPen pen(wxSystemSettings::GetSystemColour(
1149 wxSYS_COLOUR_BTNSHADOW ), 1, wxSOLID);
1150
1151 dc->SetPen( pen );
1152 dc->DrawLine( x+w-m_corner, y, x+w-1, y+h ); // right (inner)
1153 dc->DrawRectangle( x+1, y+h-1, w-2, 1 ); // bottom (inner)
1154
1155 dc->SetPen( *wxWHITE_PEN );
1156 dc->DrawRectangle( x, y, w-m_corner+1, 1 ); // top (outer)
1157 dc->DrawRectangle( x, y, 1, h ); // left (outer)
1158 dc->DrawLine( x, y+h-1, x+1, y+h-1 );
1159 dc->DrawLine( x+w-1, y, x+w-1, y+1 );
1160 #endif
1161 }
1162
1163 // shift the DC origin to match the position of the main window horz
1164 // scrollbar: this allows us to always use logical coords
1165 void wxTreeListHeaderWindow::AdjustDC(wxDC& dc)
1166 {
1167 int xpix;
1168 m_owner->GetScrollPixelsPerUnit( &xpix, NULL );
1169
1170 int x;
1171 m_owner->GetViewStart( &x, NULL );
1172
1173 // account for the horz scrollbar offset
1174 dc.SetDeviceOrigin( -x * xpix, 0 );
1175 }
1176
1177 void wxTreeListHeaderWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
1178 {
1179 static const int HEADER_OFFSET_X = 1, HEADER_OFFSET_Y = 1;
1180 #ifdef __WXGTK__
1181 wxClientDC dc( this );
1182 #else
1183 wxPaintDC dc( this );
1184 #endif
1185
1186 PrepareDC( dc );
1187 AdjustDC( dc );
1188
1189 dc.BeginDrawing();
1190
1191 dc.SetFont( GetFont() );
1192
1193 // width and height of the entire header window
1194 int w, h;
1195 GetClientSize( &w, &h );
1196 m_owner->CalcUnscrolledPosition(w, 0, &w, NULL);
1197
1198 dc.SetBackgroundMode(wxTRANSPARENT);
1199
1200 // do *not* use the listctrl colour for headers - one day we will have a
1201 // function to set it separately
1202 //dc.SetTextForeground( *wxBLACK );
1203 dc.SetTextForeground(wxSystemSettings::
1204 GetSystemColour( wxSYS_COLOUR_WINDOWTEXT ));
1205
1206 int x = HEADER_OFFSET_X;
1207
1208 int numColumns = GetColumnCount();
1209 for ( int i = 0; i < numColumns && x < w; i++ )
1210 {
1211 if (!GetColumnShown (i)) continue;
1212
1213 wxTreeListColumnInfo& column = GetColumn(i);
1214 int wCol = column.GetWidth();
1215
1216 // the width of the rect to draw: make it smaller to fit entirely
1217 // inside the column rect
1218 int cw = wCol - 2;
1219
1220 dc.SetPen( *wxWHITE_PEN );
1221
1222 DoDrawRect( &dc, x, HEADER_OFFSET_Y, cw, h-2 );
1223
1224 // if we have an image, draw it on the right of the label
1225 int image = column.GetImage(); //item.m_image;
1226 int ix = -2, iy = 0;
1227 wxImageList* imageList = m_owner->GetImageList();
1228 if(image != -1) {
1229 if(imageList) {
1230 imageList->GetSize(image, ix, iy);
1231 }
1232 //else: ignore the column image
1233 }
1234
1235 // extra margins around the text label
1236 static const int EXTRA_WIDTH = 3;
1237 static const int EXTRA_HEIGHT = 4;
1238
1239 int text_width = 0;
1240 int text_x = x;
1241 int image_offset = cw - ix - 1;
1242
1243 switch(column.GetAlignment()) {
1244 case wxTL_ALIGN_LEFT:
1245 text_x += EXTRA_WIDTH;
1246 cw -= ix + 2;
1247 break;
1248 case wxTL_ALIGN_RIGHT:
1249 dc.GetTextExtent(column.GetText(), &text_width, NULL);
1250 text_x += cw - text_width - EXTRA_WIDTH;
1251 image_offset = 0;
1252 break;
1253 case wxTL_ALIGN_CENTER:
1254 dc.GetTextExtent(column.GetText(), &text_width, NULL);
1255 text_x += (cw - text_width)/2 + ix + 2;
1256 image_offset = (cw - text_width - ix - 2)/2;
1257 break;
1258 }
1259
1260 // draw the image
1261 if(image != -1 && imageList) {
1262 imageList->Draw(image, dc, x + image_offset/*cw - ix - 1*/,
1263 HEADER_OFFSET_Y + (h - 4 - iy)/2,
1264 wxIMAGELIST_DRAW_TRANSPARENT);
1265 }
1266
1267 // draw the text clipping it so that it doesn't overwrite the column
1268 // boundary
1269 wxDCClipper clipper(dc, x, HEADER_OFFSET_Y, cw, h - 4 );
1270
1271 dc.DrawText( column.GetText(),
1272 text_x, HEADER_OFFSET_Y + EXTRA_HEIGHT );
1273
1274 x += wCol;
1275 }
1276
1277 int more_w = m_owner->GetSize().x - x;
1278 if (more_w > 0)
1279 {
1280 DoDrawRect( &dc, x, HEADER_OFFSET_Y, more_w, h-2 );
1281 }
1282
1283
1284 dc.EndDrawing();
1285 }
1286
1287 void wxTreeListHeaderWindow::DrawCurrent()
1288 {
1289 int x1 = m_currentX;
1290 int y1 = 0;
1291 ClientToScreen( &x1, &y1 );
1292
1293 int x2 = m_currentX-1;
1294 #ifdef __WXMSW__
1295 ++x2; // but why ?
1296 #endif
1297 int y2 = 0;
1298 m_owner->GetClientSize( NULL, &y2 );
1299 m_owner->ClientToScreen( &x2, &y2 );
1300
1301 wxScreenDC dc;
1302 dc.SetLogicalFunction( wxINVERT );
1303 dc.SetPen( wxPen( *wxBLACK, 2, wxSOLID ) );
1304 dc.SetBrush( *wxTRANSPARENT_BRUSH );
1305
1306 AdjustDC(dc);
1307
1308 dc.DrawLine( x1, y1, x2, y2 );
1309
1310 dc.SetLogicalFunction( wxCOPY );
1311
1312 dc.SetPen( wxNullPen );
1313 dc.SetBrush( wxNullBrush );
1314 }
1315
1316 void wxTreeListHeaderWindow::OnMouse( wxMouseEvent &event )
1317 {
1318 // we want to work with logical coords
1319 int x;
1320 m_owner->CalcUnscrolledPosition(event.GetX(), 0, &x, NULL);
1321 int y = event.GetY();
1322
1323 if (m_isDragging)
1324 {
1325 SendListEvent(wxEVT_COMMAND_LIST_COL_DRAGGING,
1326 event.GetPosition());
1327
1328 // we don't draw the line beyond our window, but we allow dragging it
1329 // there
1330 int w = 0;
1331 GetClientSize( &w, NULL );
1332 m_owner->CalcUnscrolledPosition(w, 0, &w, NULL);
1333 w -= 6;
1334
1335 // erase the line if it was drawn
1336 if ( m_currentX < w )
1337 DrawCurrent();
1338
1339 if (event.ButtonUp())
1340 {
1341 ReleaseMouse();
1342 m_isDragging = FALSE;
1343 m_dirty = TRUE;
1344 SetColumnWidth( m_column, m_currentX - m_minX );
1345 Refresh();
1346 SendListEvent(wxEVT_COMMAND_LIST_COL_END_DRAG,
1347 event.GetPosition());
1348 }
1349 else
1350 {
1351 if (x > m_minX + 7)
1352 m_currentX = x;
1353 else
1354 m_currentX = m_minX + 7;
1355
1356 // draw in the new location
1357 if ( m_currentX < w )
1358 DrawCurrent();
1359 }
1360 }
1361 else // not dragging
1362 {
1363 m_minX = 0;
1364 bool hit_border = FALSE;
1365
1366 // end of the current column
1367 int xpos = 0;
1368
1369 // find the column where this event occured
1370 int countCol = GetColumnCount();
1371 for (int col = 0; col < countCol; col++)
1372 {
1373 if (!GetColumnShown (col)) continue;
1374 xpos += GetColumnWidth (col);
1375 m_column = col;
1376
1377 if ( (abs(x-xpos) < 3) && (y < 22) )
1378 {
1379 // near the column border
1380 hit_border = TRUE;
1381 break;
1382 }
1383
1384 if ( x < xpos )
1385 {
1386 // inside the column
1387 break;
1388 }
1389
1390 m_minX = xpos;
1391 }
1392
1393 if (event.LeftDown() || event.RightUp())
1394 {
1395 if (hit_border && event.LeftDown())
1396 {
1397 m_isDragging = TRUE;
1398 m_currentX = x;
1399 DrawCurrent();
1400 CaptureMouse();
1401 SendListEvent(wxEVT_COMMAND_LIST_COL_BEGIN_DRAG,
1402 event.GetPosition());
1403 }
1404 else // click on a column
1405 {
1406 SendListEvent( event.LeftDown()
1407 ? wxEVT_COMMAND_LIST_COL_CLICK
1408 : wxEVT_COMMAND_LIST_COL_RIGHT_CLICK,
1409 event.GetPosition());
1410 }
1411 }
1412 else if (event.Moving())
1413 {
1414 bool setCursor;
1415 if (hit_border)
1416 {
1417 setCursor = m_currentCursor == wxSTANDARD_CURSOR;
1418 m_currentCursor = m_resizeCursor;
1419 }
1420 else
1421 {
1422 setCursor = m_currentCursor != wxSTANDARD_CURSOR;
1423 m_currentCursor = wxSTANDARD_CURSOR;
1424 }
1425
1426 if ( setCursor )
1427 SetCursor(*m_currentCursor);
1428 }
1429 }
1430 }
1431
1432 void wxTreeListHeaderWindow::OnSetFocus( wxFocusEvent &WXUNUSED(event) )
1433 {
1434 m_owner->SetFocus();
1435 }
1436
1437 void wxTreeListHeaderWindow::SendListEvent(wxEventType type, wxPoint pos)
1438 {
1439 wxWindow *parent = GetParent();
1440 wxListEvent le( type, parent->GetId() );
1441 le.SetEventObject( parent );
1442 le.m_pointDrag = pos;
1443
1444 // the position should be relative to the parent window, not
1445 // this one for compatibility with MSW and common sense: the
1446 // user code doesn't know anything at all about this header
1447 // window, so why should it get positions relative to it?
1448 le.m_pointDrag.y -= GetSize().y;
1449
1450 le.m_col = m_column;
1451 parent->GetEventHandler()->ProcessEvent( le );
1452 }
1453
1454 void wxTreeListHeaderWindow::AddColumn(const wxTreeListColumnInfo& col)
1455 {
1456 m_columns.Add(col);
1457 m_total_col_width += col.GetWidth();
1458 //m_owner->GetHeaderWindow()->Refresh();
1459 //m_dirty = TRUE;
1460 m_owner->AdjustMyScrollbars();
1461 m_owner->m_dirty = TRUE;
1462 Refresh();
1463 }
1464
1465 void wxTreeListHeaderWindow::SetColumnWidth(size_t column, size_t width)
1466 {
1467 if(column < GetColumnCount()) {
1468 m_total_col_width -= m_columns[column].GetWidth();
1469 m_columns[column].SetWidth(width);
1470 m_total_col_width += width;
1471 m_owner->AdjustMyScrollbars();
1472 m_owner->m_dirty = TRUE;
1473 //m_dirty = TRUE;
1474 Refresh();
1475 }
1476 }
1477
1478
1479 void wxTreeListHeaderWindow::InsertColumn(size_t before,
1480 const wxTreeListColumnInfo& col)
1481 {
1482 wxCHECK_RET(before < GetColumnCount(), wxT("Invalid column index"));
1483 m_columns.Insert(col, before);
1484 m_total_col_width += col.GetWidth();
1485 //m_dirty = TRUE;
1486 //m_owner->GetHeaderWindow()->Refresh();
1487 m_owner->AdjustMyScrollbars();
1488 m_owner->m_dirty = TRUE;
1489 Refresh();
1490 }
1491
1492 void wxTreeListHeaderWindow::RemoveColumn(size_t column)
1493 {
1494 wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
1495 m_total_col_width -= m_columns[column].GetWidth();
1496 m_columns.RemoveAt(column);
1497 //m_dirty = TRUE;
1498 m_owner->AdjustMyScrollbars();
1499 m_owner->m_dirty = TRUE;
1500 Refresh();
1501 }
1502
1503 void wxTreeListHeaderWindow::SetColumn(size_t column,
1504 const wxTreeListColumnInfo& info)
1505 {
1506 wxCHECK_RET(column < GetColumnCount(), wxT("Invalid column"));
1507 size_t w = m_columns[column].GetWidth();
1508 m_columns[column] = info;
1509 //m_owner->GetHeaderWindow()->Refresh();
1510 //m_dirty = TRUE;
1511 if(w != info.GetWidth()) {
1512 m_total_col_width += info.GetWidth() - w;
1513 m_owner->AdjustMyScrollbars();
1514 m_owner->m_dirty = TRUE;
1515 }
1516 Refresh();
1517 }
1518
1519 // ---------------------------------------------------------------------------
1520 // wxTreeListItem
1521 // ---------------------------------------------------------------------------
1522
1523 wxTreeListItem::wxTreeListItem(wxTreeListMainWindow *owner,
1524 wxTreeListItem *parent,
1525 const wxArrayString& text,
1526 int image, int selImage,
1527 wxTreeItemData *data)
1528 : m_text(text)
1529 {
1530 m_images[wxTreeItemIcon_Normal] = image;
1531 m_images[wxTreeItemIcon_Selected] = selImage;
1532 m_images[wxTreeItemIcon_Expanded] = NO_IMAGE;
1533 m_images[wxTreeItemIcon_SelectedExpanded] = NO_IMAGE;
1534
1535 m_data = data;
1536 m_x = m_y = 0;
1537
1538 m_isCollapsed = TRUE;
1539 m_hasHilight = FALSE;
1540 m_hasPlus = FALSE;
1541 m_isBold = FALSE;
1542
1543 m_owner = owner;
1544
1545 m_parent = parent;
1546
1547 m_attr = (wxTreeItemAttr *)NULL;
1548 m_ownsAttr = FALSE;
1549
1550 // We don't know the height here yet.
1551 m_width = 0;
1552 m_height = 0;
1553 }
1554
1555 wxTreeListItem::~wxTreeListItem()
1556 {
1557 delete m_data;
1558
1559 if (m_ownsAttr) delete m_attr;
1560
1561 wxASSERT_MSG( m_children.IsEmpty(),
1562 wxT("please call DeleteChildren() before deleting the item") );
1563 }
1564
1565 void wxTreeListItem::DeleteChildren(wxTreeListMainWindow *tree)
1566 {
1567 size_t count = m_children.Count();
1568 for ( size_t n = 0; n < count; n++ )
1569 {
1570 wxTreeListItem *child = m_children[n];
1571 if (tree)
1572 tree->SendDeleteEvent(child);
1573
1574 child->DeleteChildren(tree);
1575 delete child;
1576 }
1577
1578 m_children.Empty();
1579 }
1580
1581 void wxTreeListItem::SetText( const wxString &text )
1582 {
1583 if(m_text.GetCount() > 0) m_text[0] = text;
1584 else {
1585 m_text.Add(text);
1586 }
1587 }
1588
1589 size_t wxTreeListItem::GetChildrenCount(bool recursively) const
1590 {
1591 size_t count = m_children.Count();
1592 if ( !recursively )
1593 return count;
1594
1595 size_t total = count;
1596 for (size_t n = 0; n < count; ++n)
1597 {
1598 total += m_children[n]->GetChildrenCount();
1599 }
1600
1601 return total;
1602 }
1603
1604 void wxTreeListItem::GetSize( int &x, int &y,
1605 const wxTreeListMainWindow *theButton )
1606 {
1607 int bottomY=m_y+theButton->GetLineHeight(this);
1608 if ( y < bottomY ) y = bottomY;
1609 int width = m_x + m_width;
1610 if ( x < width ) x = width;
1611
1612 if (IsExpanded())
1613 {
1614 size_t count = m_children.Count();
1615 for ( size_t n = 0; n < count; ++n )
1616 {
1617 m_children[n]->GetSize( x, y, theButton );
1618 }
1619 }
1620 }
1621
1622 wxTreeListItem *wxTreeListItem::HitTest(const wxPoint& point,
1623 const wxTreeListMainWindow *theCtrl,
1624 int &flags,
1625 int level)
1626 {
1627 // for a hidden root node, don't evaluate it, but do evaluate children
1628 if (!(theCtrl->HasFlag(wxTR_HIDE_ROOT) && (level == 0)))
1629 {
1630 // evaluate the item
1631 int h = theCtrl->GetLineHeight(this);
1632 if ((point.y > m_y) && (point.y < m_y + h))
1633 {
1634 // check for above/below middle
1635 int y_mid = m_y + h/2;
1636 if (point.y < y_mid )
1637 flags |= wxTREE_HITTEST_ONITEMUPPERPART;
1638 else
1639 flags |= wxTREE_HITTEST_ONITEMLOWERPART;
1640
1641 // check for button hit
1642 if (HasPlus() && theCtrl->HasButtons()) {
1643 int bntX = m_x - theCtrl->m_btnWidth2;
1644 int bntY = y_mid - theCtrl->m_btnHeight2;
1645 if ((point.x > bntX) && (point.x < (bntX + theCtrl->m_btnWidth)) &&
1646 (point.y > bntY) && (point.y < (bntY + theCtrl->m_btnHeight))) {
1647 flags |= wxTREE_HITTEST_ONITEMBUTTON;
1648 return this;
1649 }
1650 }
1651
1652 // check for image hit
1653 if (theCtrl->m_imgWidth > 0) {
1654 int imgX = m_x - theCtrl->m_imgWidth2;
1655 int imgY = y_mid - theCtrl->m_imgHeight2;
1656 if ((point.x >= imgX) && (point.x <= (imgX + theCtrl->m_imgWidth)) &&
1657 (point.y >= imgY) && (point.y <= (imgY + theCtrl->m_imgHeight))) {
1658 flags |= wxTREE_HITTEST_ONITEMICON;
1659 return this;
1660 }
1661 }
1662
1663 // check for label hit
1664 int lblX = m_x - theCtrl->m_imgWidth2 + theCtrl->m_imgWidth + MARGIN;
1665 if ((point.x >= lblX) && (point.x <= (m_x + m_width)) &&
1666 (point.y >= m_y) && (point.y <= (m_y + h))) {
1667 flags |= wxTREE_HITTEST_ONITEMLABEL;
1668 return this;
1669 }
1670
1671 // else check for indent
1672 if (point.x < m_x) {
1673 flags |= wxTREE_HITTEST_ONITEMINDENT;
1674 return this;
1675 }
1676
1677 // else check for item right???
1678 if (point.x > m_x + m_width) {
1679 flags |= wxTREE_HITTEST_ONITEMRIGHT;
1680 return this;
1681 }
1682
1683 }
1684
1685 // if children are expanded, fall through to evaluate them
1686 if (m_isCollapsed) return (wxTreeListItem*) NULL;
1687 }
1688
1689 // evaluate children
1690 size_t count = m_children.Count();
1691 for ( size_t n = 0; n < count; n++ )
1692 {
1693 wxTreeListItem *res = m_children[n]->HitTest(point, theCtrl,
1694 flags, level + 1);
1695 if ( res != NULL )
1696 return res;
1697 }
1698
1699 return (wxTreeListItem*) NULL;
1700 }
1701
1702 // ALB
1703 wxTreeListItem *wxTreeListItem::HitTest(const wxPoint& point,
1704 const wxTreeListMainWindow *theCtrl,
1705 int &flags, int& column, int level)
1706 {
1707 column = theCtrl->GetMainColumn(); //-1;
1708
1709 wxTreeListItem* res = HitTest(point, theCtrl, flags, level);
1710 if(!res) {
1711 column = -1;
1712 return res;
1713 }
1714
1715 wxTreeListHeaderWindow* header_win = theCtrl->m_owner->GetHeaderWindow();
1716 if (point.x >= header_win->GetWidth())
1717 column = -1;
1718 else if(flags & wxTREE_HITTEST_ONITEMINDENT) {
1719 int x = 0;
1720 for(size_t i = 0; i < theCtrl->GetMainColumn(); ++i) {
1721 if (!header_win->GetColumnShown(i)) continue;
1722 int w = header_win->GetColumnWidth(i);
1723 if(point.x >= x && point.x < x+w) {
1724 flags ^= wxTREE_HITTEST_ONITEMINDENT;
1725 flags |= wxTREE_HITTEST_ONITEMCOLUMN;
1726 column = i;
1727 return res;
1728 }
1729 }
1730 }
1731 else if(flags & wxTREE_HITTEST_ONITEMRIGHT) {
1732 int x = 0;
1733 size_t i;
1734 for(i = 0; i < theCtrl->GetMainColumn()+1; ++i) {
1735 if (!header_win->GetColumnShown(i)) continue;
1736 x += header_win->GetColumnWidth(i);
1737 }
1738 for(i = theCtrl->GetMainColumn()+1; i < theCtrl->GetColumnCount(); ++i) {
1739 if (!header_win->GetColumnShown(i)) continue;
1740 int w = header_win->GetColumnWidth(i);
1741 if(point.x >= x && point.x < x+w) {
1742 flags ^= wxTREE_HITTEST_ONITEMRIGHT;
1743 flags |= wxTREE_HITTEST_ONITEMCOLUMN;
1744 column = i;
1745 return res;
1746 }
1747 }
1748 }
1749
1750 return res;
1751 }
1752
1753
1754 int wxTreeListItem::GetCurrentImage() const
1755 {
1756 int image = NO_IMAGE;
1757 if ( IsExpanded() )
1758 {
1759 if ( IsSelected() )
1760 {
1761 image = GetImage(wxTreeItemIcon_SelectedExpanded);
1762 }
1763
1764 if ( image == NO_IMAGE )
1765 {
1766 // we usually fall back to the normal item, but try just the
1767 // expanded one (and not selected) first in this case
1768 image = GetImage(wxTreeItemIcon_Expanded);
1769 }
1770 }
1771 else // not expanded
1772 {
1773 if ( IsSelected() )
1774 image = GetImage(wxTreeItemIcon_Selected);
1775 }
1776
1777 // maybe it doesn't have the specific image we want,
1778 // try the default one instead
1779 if ( image == NO_IMAGE ) image = GetImage();
1780
1781 return image;
1782 }
1783
1784 // ---------------------------------------------------------------------------
1785 // wxTreeListMainWindow implementation
1786 // ---------------------------------------------------------------------------
1787
1788 IMPLEMENT_DYNAMIC_CLASS(wxTreeListMainWindow, wxScrolledWindow)
1789
1790 BEGIN_EVENT_TABLE(wxTreeListMainWindow, wxScrolledWindow)
1791 EVT_PAINT (wxTreeListMainWindow::OnPaint)
1792 EVT_MOUSE_EVENTS (wxTreeListMainWindow::OnMouse)
1793 EVT_CHAR (wxTreeListMainWindow::OnChar)
1794 EVT_SET_FOCUS (wxTreeListMainWindow::OnSetFocus)
1795 EVT_KILL_FOCUS (wxTreeListMainWindow::OnKillFocus)
1796 EVT_IDLE (wxTreeListMainWindow::OnIdle)
1797 //? EVT_SIZE (wxTreeListMainWindow::OnSize)
1798 EVT_SCROLLWIN (wxTreeListMainWindow::OnScroll)
1799 END_EVENT_TABLE()
1800
1801
1802 // ---------------------------------------------------------------------------
1803 // construction/destruction
1804 // ---------------------------------------------------------------------------
1805
1806 void wxTreeListMainWindow::Init()
1807 {
1808 m_current = m_key_current = m_anchor = (wxTreeListItem *) NULL;
1809 m_hasFocus = FALSE;
1810 m_dirty = FALSE;
1811
1812 m_lineHeight = LINEHEIGHT;
1813 m_indent = MININDENT; // min. indent
1814 m_linespacing = 4;
1815
1816 m_hilightBrush = new wxBrush
1817 (
1818 wxSystemSettings::GetSystemColour
1819 (
1820 wxSYS_COLOUR_HIGHLIGHT
1821 ),
1822 wxSOLID
1823 );
1824
1825 m_hilightUnfocusedBrush = new wxBrush
1826 (
1827 wxSystemSettings::GetSystemColour
1828 (
1829 wxSYS_COLOUR_BTNSHADOW
1830 ),
1831 wxSOLID
1832 );
1833
1834 m_imageListNormal = m_imageListButtons =
1835 m_imageListState = (wxImageList *) NULL;
1836 m_ownsImageListNormal = m_ownsImageListButtons =
1837 m_ownsImageListState = FALSE;
1838
1839 m_dragCount = 0;
1840 m_isDragging = FALSE;
1841 m_dropTarget = m_oldSelection = (wxTreeListItem *)NULL;
1842
1843 m_renameTimer = new wxTreeListRenameTimer( this );
1844 m_lastOnSame = FALSE;
1845
1846 m_findTimer = new wxTimer (this, -1);
1847
1848 m_normalFont = wxSystemSettings::GetSystemFont( wxSYS_DEFAULT_GUI_FONT );
1849 m_boldFont = wxFont( m_normalFont.GetPointSize(),
1850 m_normalFont.GetFamily(),
1851 m_normalFont.GetStyle(),
1852 wxBOLD,
1853 m_normalFont.GetUnderlined());
1854 }
1855
1856
1857 static const int HEADER_HEIGHT = 23;
1858
1859 bool wxTreeListMainWindow::Create(wxTreeListCtrl *parent,
1860 wxWindowID id,
1861 const wxPoint& pos,
1862 const wxSize& size,
1863 long style,
1864 const wxValidator &validator,
1865 const wxString& name )
1866 {
1867 #ifdef __WXMAC__
1868 int major,minor;
1869 wxGetOsVersion( &major, &minor );
1870
1871 if (style & wxTR_HAS_BUTTONS) style |= wxTR_MAC_BUTTONS;
1872 if (style & wxTR_HAS_BUTTONS) style &= ~wxTR_HAS_BUTTONS;
1873 style &= ~wxTR_LINES_AT_ROOT;
1874 style |= wxTR_NO_LINES;
1875 if (major < 10)
1876 style |= wxTR_ROW_LINES;
1877 #endif
1878
1879 wxScrolledWindow::Create( parent, id, pos, size,
1880 style|wxHSCROLL|wxVSCROLL, name );
1881
1882 #if wxUSE_VALIDATORS
1883 SetValidator( validator );
1884 #endif
1885
1886 SetBackgroundColour( wxSystemSettings::GetSystemColour( wxSYS_COLOUR_LISTBOX ) );
1887
1888 #ifdef __WXMSW__
1889 {
1890 int i, j;
1891 wxBitmap bmp(8, 8);
1892 wxMemoryDC bdc;
1893 bdc.SelectObject(bmp);
1894 bdc.SetPen(*wxGREY_PEN);
1895 bdc.DrawRectangle(-1, -1, 10, 10);
1896 for (i = 0; i < 8; i++) {
1897 for (j = 0; j < 8; j++) {
1898 if (!((i + j) & 1)) {
1899 bdc.DrawPoint(i, j);
1900 }
1901 }
1902 }
1903
1904 m_dottedPen = wxPen(bmp, 1);
1905 }
1906 #else
1907 //m_dottedPen = wxPen( *wxGREY_PEN, 1, wxDOT ); // too slow under XFree86
1908 m_dottedPen = wxPen( wxT("grey"), 0, 0 ); // Bitmap based pen is not supported by GTK!
1909 #endif
1910
1911 // ALB
1912 m_owner = parent;
1913 m_main_column = 0;
1914
1915 return TRUE;
1916 }
1917
1918 wxTreeListMainWindow::~wxTreeListMainWindow()
1919 {
1920 delete m_hilightBrush;
1921 delete m_hilightUnfocusedBrush;
1922
1923 DeleteAllItems();
1924
1925 delete m_renameTimer;
1926 delete m_findTimer;
1927 if (m_ownsImageListNormal) delete m_imageListNormal;
1928 if (m_ownsImageListState) delete m_imageListState;
1929 if (m_ownsImageListButtons) delete m_imageListButtons;
1930 }
1931
1932
1933
1934 //-----------------------------------------------------------------------------
1935 // accessors
1936 //-----------------------------------------------------------------------------
1937
1938 size_t wxTreeListMainWindow::GetCount() const
1939 {
1940 return m_anchor == NULL ? 0u : m_anchor->GetChildrenCount();
1941 }
1942
1943 void wxTreeListMainWindow::SetIndent(unsigned int indent)
1944 {
1945 m_indent = indent;
1946 m_dirty = TRUE;
1947 }
1948
1949 void wxTreeListMainWindow::SetLineSpacing(unsigned int spacing)
1950 {
1951 m_linespacing = spacing;
1952 m_dirty = TRUE;
1953 CalculateLineHeight();
1954 }
1955
1956 size_t wxTreeListMainWindow::GetChildrenCount(const wxTreeItemId& item,
1957 bool recursively)
1958 {
1959 wxCHECK_MSG( item.IsOk(), 0u, wxT("invalid tree item") );
1960
1961 return ((wxTreeListItem*) item.m_pItem)->GetChildrenCount(recursively);
1962 }
1963
1964 void wxTreeListMainWindow::SetWindowStyle(const long styles)
1965 {
1966 // right now, just sets the styles. Eventually, we may
1967 // want to update the inherited styles, but right now
1968 // none of the parents has updatable styles
1969 m_windowStyle = styles;
1970 m_dirty = TRUE;
1971 }
1972
1973 //-----------------------------------------------------------------------------
1974 // functions to work with tree items
1975 //-----------------------------------------------------------------------------
1976
1977 int wxTreeListMainWindow::GetItemImage(const wxTreeItemId& item, size_t column,
1978 wxTreeItemIcon which) const
1979 {
1980 wxCHECK_MSG( item.IsOk(), -1, wxT("invalid tree item") );
1981
1982 return ((wxTreeListItem*) item.m_pItem)->GetImage(column, which);
1983 }
1984
1985 wxTreeItemData *wxTreeListMainWindow::GetItemData(const wxTreeItemId& item)
1986 const
1987 {
1988 wxCHECK_MSG( item.IsOk(), NULL, wxT("invalid tree item") );
1989
1990 return ((wxTreeListItem*) item.m_pItem)->GetData();
1991 }
1992
1993 bool wxTreeListMainWindow::GetItemBold(const wxTreeItemId& item) const
1994 {
1995 wxCHECK_MSG(item.IsOk(), FALSE, wxT("invalid tree item"));
1996 return ((wxTreeListItem *)item.m_pItem)->IsBold();
1997 }
1998
1999 wxColour wxTreeListMainWindow::GetItemTextColour(const wxTreeItemId& item)
2000 const
2001 {
2002 wxCHECK_MSG( item.IsOk(), wxNullColour, wxT("invalid tree item") );
2003
2004 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2005 return pItem->Attr().GetTextColour();
2006 }
2007
2008 wxColour wxTreeListMainWindow::GetItemBackgroundColour(
2009 const wxTreeItemId& item) const
2010 {
2011 wxCHECK_MSG( item.IsOk(), wxNullColour, wxT("invalid tree item") );
2012
2013 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2014 return pItem->Attr().GetBackgroundColour();
2015 }
2016
2017 wxFont wxTreeListMainWindow::GetItemFont(const wxTreeItemId& item) const
2018 {
2019 wxCHECK_MSG( item.IsOk(), wxNullFont, wxT("invalid tree item") );
2020
2021 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2022 return pItem->Attr().GetFont();
2023 }
2024
2025
2026
2027 void wxTreeListMainWindow::SetItemImage(const wxTreeItemId& item,
2028 size_t column,
2029 int image, wxTreeItemIcon which)
2030 {
2031 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2032
2033 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2034 pItem->SetImage(column, image, which);
2035
2036 wxClientDC dc(this);
2037 CalculateSize(pItem, dc);
2038 RefreshLine(pItem);
2039 }
2040
2041 void wxTreeListMainWindow::SetItemData(const wxTreeItemId& item,
2042 wxTreeItemData *data)
2043 {
2044 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2045
2046 ((wxTreeListItem*) item.m_pItem)->SetData(data);
2047 }
2048
2049 void wxTreeListMainWindow::SetItemHasChildren(const wxTreeItemId& item,
2050 bool has)
2051 {
2052 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2053
2054 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2055 pItem->SetHasPlus(has);
2056 RefreshLine(pItem);
2057 }
2058
2059 void wxTreeListMainWindow::SetItemBold(const wxTreeItemId& item, bool bold)
2060 {
2061 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2062
2063 // avoid redrawing the tree if no real change
2064 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2065 if ( pItem->IsBold() != bold )
2066 {
2067 pItem->SetBold(bold);
2068 RefreshLine(pItem);
2069 }
2070 }
2071
2072 void wxTreeListMainWindow::SetItemTextColour(const wxTreeItemId& item,
2073 const wxColour& col)
2074 {
2075 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2076
2077 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2078 pItem->Attr().SetTextColour(col);
2079 RefreshLine(pItem);
2080 }
2081
2082 void wxTreeListMainWindow::SetItemBackgroundColour(const wxTreeItemId& item,
2083 const wxColour& col)
2084 {
2085 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2086
2087 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2088 pItem->Attr().SetBackgroundColour(col);
2089 RefreshLine(pItem);
2090 }
2091
2092 void wxTreeListMainWindow::SetItemFont(const wxTreeItemId& item,
2093 const wxFont& font)
2094 {
2095 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2096
2097 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2098 pItem->Attr().SetFont(font);
2099 RefreshLine(pItem);
2100 }
2101
2102 bool wxTreeListMainWindow::SetFont( const wxFont &font )
2103 {
2104 wxScrolledWindow::SetFont(font);
2105
2106 m_normalFont = font ;
2107 m_boldFont = wxFont( m_normalFont.GetPointSize(),
2108 m_normalFont.GetFamily(),
2109 m_normalFont.GetStyle(),
2110 wxBOLD,
2111 m_normalFont.GetUnderlined());
2112
2113 return TRUE;
2114 }
2115
2116
2117 // ----------------------------------------------------------------------------
2118 // item status inquiries
2119 // ----------------------------------------------------------------------------
2120
2121 bool wxTreeListMainWindow::IsVisible(const wxTreeItemId& item) const
2122 {
2123 wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
2124
2125 // An item is only visible if it's not a descendant of a collapsed item
2126 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
2127 wxTreeListItem* parent = pItem->GetItemParent();
2128 while (parent)
2129 {
2130 if (!parent->IsExpanded())
2131 return FALSE;
2132 parent = parent->GetItemParent();
2133 }
2134
2135 int startX, startY;
2136 GetViewStart(& startX, & startY);
2137
2138 wxSize clientSize = GetClientSize();
2139
2140 wxRect rect;
2141 if (!GetBoundingRect(item, rect))
2142 return FALSE;
2143 if (rect.GetWidth() == 0 || rect.GetHeight() == 0)
2144 return FALSE;
2145 if (rect.GetBottom() < 0 || rect.GetTop() > clientSize.y)
2146 return FALSE;
2147 if (rect.GetRight() < 0 || rect.GetLeft() > clientSize.x)
2148 return FALSE;
2149
2150 return TRUE;
2151 }
2152
2153 bool wxTreeListMainWindow::ItemHasChildren(const wxTreeItemId& item) const
2154 {
2155 wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
2156
2157 // consider that the item does have children if it has the "+" button: it
2158 // might not have them (if it had never been expanded yet) but then it
2159 // could have them as well and it's better to err on this side rather than
2160 // disabling some operations which are restricted to the items with
2161 // children for an item which does have them
2162 return ((wxTreeListItem*) item.m_pItem)->HasPlus();
2163 }
2164
2165 bool wxTreeListMainWindow::IsExpanded(const wxTreeItemId& item) const
2166 {
2167 wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
2168
2169 return ((wxTreeListItem*) item.m_pItem)->IsExpanded();
2170 }
2171
2172 bool wxTreeListMainWindow::IsSelected(const wxTreeItemId& item) const
2173 {
2174 wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
2175
2176 return ((wxTreeListItem*) item.m_pItem)->IsSelected();
2177 }
2178
2179 bool wxTreeListMainWindow::IsBold(const wxTreeItemId& item) const
2180 {
2181 wxCHECK_MSG( item.IsOk(), FALSE, wxT("invalid tree item") );
2182
2183 return ((wxTreeListItem*) item.m_pItem)->IsBold();
2184 }
2185
2186 // ----------------------------------------------------------------------------
2187 // navigation
2188 // ----------------------------------------------------------------------------
2189
2190 wxTreeItemId wxTreeListMainWindow::GetItemParent(const wxTreeItemId& item) const
2191 {
2192 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2193
2194 return ((wxTreeListItem*) item.m_pItem)->GetItemParent();
2195 }
2196
2197 #if !wxCHECK_VERSION(2, 5, 0)
2198 wxTreeItemId wxTreeListMainWindow::GetFirstChild(const wxTreeItemId& item,
2199 long& cookie) const
2200 #else
2201 wxTreeItemId wxTreeListMainWindow::GetFirstChild(const wxTreeItemId& item,
2202 wxTreeItemIdValue& cookie) const
2203 #endif
2204 {
2205 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2206
2207 cookie = 0;
2208 return GetNextChild(item, cookie);
2209 }
2210
2211 #if !wxCHECK_VERSION(2, 5, 0)
2212 wxTreeItemId wxTreeListMainWindow::GetNextChild(const wxTreeItemId& item,
2213 long& cookie) const
2214 #else
2215 wxTreeItemId wxTreeListMainWindow::GetNextChild(const wxTreeItemId& item,
2216 wxTreeItemIdValue& cookie) const
2217 #endif
2218 {
2219 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2220
2221 wxArrayTreeListItems& children = ((wxTreeListItem*) item.m_pItem)->GetChildren();
2222
2223 // it's ok to cast cookie to size_t, we never have indices big enough to
2224 // overflow "void *"
2225 size_t *pIndex = (size_t *)&cookie;
2226 if ( *pIndex < children.Count() )
2227 {
2228 return children.Item((*pIndex)++);
2229 }
2230 else
2231 {
2232 // there are no more of them
2233 return wxTreeItemId();
2234 }
2235 }
2236
2237 #if !wxCHECK_VERSION(2, 5, 0)
2238 wxTreeItemId wxTreeListMainWindow::GetPrevChild(const wxTreeItemId& item,
2239 long& cookie) const
2240 #else
2241 wxTreeItemId wxTreeListMainWindow::GetPrevChild(const wxTreeItemId& item,
2242 wxTreeItemIdValue& cookie) const
2243 #endif
2244 {
2245 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2246
2247 wxArrayTreeListItems& children = ((wxTreeListItem*) item.m_pItem)->GetChildren();
2248
2249 // it's ok to cast cookie to size_t, we never have indices big enough to
2250 // overflow "void *"
2251 size_t *pIndex = (size_t *)&cookie;
2252 if ( *pIndex > 0 )
2253 {
2254 return children.Item(--(*pIndex));
2255 }
2256 else
2257 {
2258 // there are no more of them
2259 return wxTreeItemId();
2260 }
2261 }
2262
2263 wxTreeItemId wxTreeListMainWindow::GetLastChild(const wxTreeItemId& item) const
2264 {
2265 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2266
2267 wxArrayTreeListItems& children = ((wxTreeListItem*) item.m_pItem)->GetChildren();
2268 return (children.IsEmpty() ? wxTreeItemId() : wxTreeItemId(children.Last()));
2269 }
2270
2271 wxTreeItemId wxTreeListMainWindow::GetNextSibling(const wxTreeItemId& item) const
2272 {
2273 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2274
2275 wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
2276 wxTreeListItem *parent = i->GetItemParent();
2277 if ( parent == NULL )
2278 {
2279 // root item doesn't have any siblings
2280 return wxTreeItemId();
2281 }
2282
2283 wxArrayTreeListItems& siblings = parent->GetChildren();
2284 int index = siblings.Index(i);
2285 wxASSERT( index != wxNOT_FOUND ); // I'm not a child of my parent?
2286
2287 size_t n = (size_t)(index + 1);
2288 return n == siblings.Count() ? wxTreeItemId() : wxTreeItemId(siblings[n]);
2289 }
2290
2291 wxTreeItemId wxTreeListMainWindow::GetPrevSibling(const wxTreeItemId& item)
2292 const
2293 {
2294 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2295
2296 wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
2297 wxTreeListItem *parent = i->GetItemParent();
2298 if ( parent == NULL )
2299 {
2300 // root item doesn't have any siblings
2301 return wxTreeItemId();
2302 }
2303
2304 wxArrayTreeListItems& siblings = parent->GetChildren();
2305 int index = siblings.Index(i);
2306 wxASSERT( index != wxNOT_FOUND ); // I'm not a child of my parent?
2307
2308 return index == 0 ? wxTreeItemId()
2309 : wxTreeItemId(siblings[(size_t)(index - 1)]);
2310 }
2311
2312 // Only for internal use right now, but should probably be public
2313 wxTreeItemId wxTreeListMainWindow::GetNext(const wxTreeItemId& item) const
2314 {
2315 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2316
2317 wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
2318
2319 // First see if there are any children.
2320 wxArrayTreeListItems& children = i->GetChildren();
2321 if (children.GetCount() > 0)
2322 {
2323 return children.Item(0);
2324 }
2325 else
2326 {
2327 // Try a sibling of this or ancestor instead
2328 wxTreeItemId p = item;
2329 wxTreeItemId toFind;
2330 do
2331 {
2332 toFind = GetNextSibling(p);
2333 p = GetItemParent(p);
2334 } while (p.IsOk() && !toFind.IsOk());
2335 return toFind;
2336 }
2337 }
2338
2339 wxTreeItemId wxTreeListMainWindow::GetFirstVisibleItem() const
2340 {
2341 wxTreeItemId id = GetRootItem();
2342 if (!id.IsOk())
2343 return id;
2344
2345 do
2346 {
2347 if (IsVisible(id))
2348 return id;
2349 id = GetNext(id);
2350 } while (id.IsOk());
2351
2352 return wxTreeItemId();
2353 }
2354
2355 wxTreeItemId wxTreeListMainWindow::GetNextVisible(const wxTreeItemId& item)
2356 const
2357 {
2358 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2359
2360 wxTreeItemId id = item;
2361 if (id.IsOk())
2362 {
2363 while (id = GetNext(id), id.IsOk())
2364 {
2365 if (IsVisible(id))
2366 return id;
2367 }
2368 }
2369 return wxTreeItemId();
2370 }
2371
2372 wxTreeItemId wxTreeListMainWindow::GetPrevVisible(const wxTreeItemId& item)
2373 const
2374 {
2375 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
2376
2377 wxFAIL_MSG(wxT("not implemented"));
2378
2379 return wxTreeItemId();
2380 }
2381
2382 // ----------------------------------------------------------------------------
2383 // operations
2384 // ----------------------------------------------------------------------------
2385
2386 wxTreeItemId wxTreeListMainWindow::DoInsertItem(const wxTreeItemId& parentId,
2387 size_t previous,
2388 const wxString& text,
2389 int image, int selImage,
2390 wxTreeItemData *data)
2391 {
2392 wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
2393 if ( !parent )
2394 {
2395 // should we give a warning here?
2396 return AddRoot(text, image, selImage, data);
2397 }
2398
2399 m_dirty = TRUE; // do this first so stuff below doesn't cause flicker
2400
2401 // ALB
2402 wxArrayString arr;
2403 arr.Alloc(GetColumnCount());
2404 for(size_t i = 0; i < GetColumnCount(); ++i) {
2405 arr.Add(wxEmptyString);
2406 }
2407 arr[m_main_column] = text;
2408 wxTreeListItem *item =
2409 new wxTreeListItem( this, parent, arr, image, selImage, data );
2410
2411 if ( data != NULL )
2412 {
2413 data->SetId((long)item);
2414 }
2415
2416 parent->Insert( item, previous );
2417
2418 return item;
2419 }
2420
2421 wxTreeItemId wxTreeListMainWindow::AddRoot(const wxString& text,
2422 int image, int selImage,
2423 wxTreeItemData *data)
2424 {
2425 wxCHECK_MSG(!m_anchor, wxTreeItemId(), wxT("tree can have only one root"));
2426 wxCHECK_MSG(GetColumnCount(), wxTreeItemId(), wxT("Add column(s) before adding the root item"));
2427
2428 m_dirty = TRUE; // do this first so stuff below doesn't cause flicker
2429
2430 // ALB
2431 wxArrayString arr;
2432 arr.Alloc(GetColumnCount());
2433 for(size_t i = 0; i < GetColumnCount(); ++i) {
2434 arr.Add(wxEmptyString);
2435 }
2436 arr[m_main_column] = text;
2437 m_anchor = new wxTreeListItem( this, (wxTreeListItem *)NULL, arr,
2438 image, selImage, data);
2439 #if 0
2440 if (HasFlag(wxTR_HIDE_ROOT))
2441 {
2442 // if root is hidden, make sure we can navigate
2443 // into children
2444 m_anchor->SetHasPlus();
2445 Expand(m_anchor);
2446 }
2447 #endif
2448 if ( data != NULL )
2449 {
2450 data->SetId((long)m_anchor);
2451 }
2452
2453 if (!HasFlag(wxTR_MULTIPLE))
2454 {
2455 m_current = m_key_current = m_anchor;
2456 m_current->SetHilight( TRUE );
2457 }
2458
2459 return m_anchor;
2460 }
2461
2462 wxTreeItemId wxTreeListMainWindow::PrependItem(const wxTreeItemId& parent,
2463 const wxString& text,
2464 int image, int selImage,
2465 wxTreeItemData *data)
2466 {
2467 return DoInsertItem(parent, 0u, text, image, selImage, data);
2468 }
2469
2470 wxTreeItemId wxTreeListMainWindow::InsertItem(const wxTreeItemId& parentId,
2471 const wxTreeItemId& idPrevious,
2472 const wxString& text,
2473 int image, int selImage,
2474 wxTreeItemData *data)
2475 {
2476 wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
2477 if ( !parent )
2478 {
2479 // should we give a warning here?
2480 return AddRoot(text, image, selImage, data);
2481 }
2482
2483 int index = parent->GetChildren().Index((wxTreeListItem*) idPrevious.m_pItem);
2484 wxASSERT_MSG( index != wxNOT_FOUND,
2485 wxT("previous item in wxTreeListMainWindow::InsertItem() is not a sibling") );
2486
2487 return DoInsertItem(parentId, (size_t)++index, text, image, selImage, data);
2488 }
2489
2490 wxTreeItemId wxTreeListMainWindow::InsertItem(const wxTreeItemId& parentId,
2491 size_t before,
2492 const wxString& text,
2493 int image, int selImage,
2494 wxTreeItemData *data)
2495 {
2496 wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
2497 if ( !parent )
2498 {
2499 // should we give a warning here?
2500 return AddRoot(text, image, selImage, data);
2501 }
2502
2503 return DoInsertItem(parentId, before, text, image, selImage, data);
2504 }
2505
2506 wxTreeItemId wxTreeListMainWindow::AppendItem(const wxTreeItemId& parentId,
2507 const wxString& text,
2508 int image, int selImage,
2509 wxTreeItemData *data)
2510 {
2511 wxTreeListItem *parent = (wxTreeListItem*) parentId.m_pItem;
2512 if ( !parent )
2513 {
2514 // should we give a warning here?
2515 return AddRoot(text, image, selImage, data);
2516 }
2517
2518 return DoInsertItem( parent, parent->GetChildren().Count(), text,
2519 image, selImage, data);
2520 }
2521
2522 void wxTreeListMainWindow::SendDeleteEvent(wxTreeListItem *item)
2523 {
2524 wxTreeEvent event( wxEVT_COMMAND_TREE_DELETE_ITEM, m_owner->GetId() );
2525 event.SetItem((long) item);
2526 event.SetEventObject( /*this*/m_owner );
2527 m_owner->ProcessEvent( event );
2528 }
2529
2530 void wxTreeListMainWindow::DeleteChildren(const wxTreeItemId& itemId)
2531 {
2532 m_dirty = TRUE; // do this first so stuff below doesn't cause flicker
2533
2534 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
2535 item->DeleteChildren(this);
2536 }
2537
2538 void wxTreeListMainWindow::Delete(const wxTreeItemId& itemId)
2539 {
2540 m_dirty = TRUE; // do this first so stuff below doesn't cause flicker
2541
2542 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
2543
2544 // don't stay with invalid m_key_current or we will crash in
2545 // the next call to OnChar()
2546 bool changeKeyCurrent = FALSE;
2547 wxTreeListItem *itemKey = m_key_current;
2548 while ( itemKey )
2549 {
2550 if ( itemKey == item )
2551 {
2552 // m_key_current is a descendant of the item being deleted
2553 changeKeyCurrent = TRUE;
2554 break;
2555 }
2556 itemKey = itemKey->GetItemParent();
2557 }
2558
2559 wxTreeListItem *parent = item->GetItemParent();
2560 if ( parent )
2561 {
2562 parent->GetChildren().Remove( item ); // remove by value
2563 }
2564
2565 if ( changeKeyCurrent )
2566 {
2567 // may be NULL or not
2568 m_key_current = parent;
2569 }
2570
2571 item->DeleteChildren(this);
2572 SendDeleteEvent(item);
2573 delete item;
2574 }
2575
2576 void wxTreeListMainWindow::DeleteAllItems()
2577 {
2578 if ( m_anchor )
2579 {
2580 m_dirty = TRUE;
2581
2582 m_anchor->DeleteChildren(this);
2583 delete m_anchor;
2584
2585 m_anchor = NULL;
2586 }
2587 }
2588
2589 void wxTreeListMainWindow::Expand(const wxTreeItemId& itemId)
2590 {
2591 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
2592
2593 wxCHECK_RET( item, _T("invalid item in wxTreeListMainWindow::Expand") );
2594
2595 if ( !item->HasPlus() )
2596 return;
2597
2598 if ( item->IsExpanded() )
2599 return;
2600
2601 wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_EXPANDING, m_owner->GetId() );
2602 event.SetItem( (long) item );
2603 event.SetEventObject( /*this*/m_owner );
2604
2605 if ( m_owner->ProcessEvent( event ) && !event.IsAllowed() )
2606 {
2607 // cancelled by program
2608 return;
2609 }
2610
2611 item->Expand();
2612 CalculatePositions();
2613
2614 RefreshSubtree(item);
2615
2616 event.SetEventType(wxEVT_COMMAND_TREE_ITEM_EXPANDED);
2617 ProcessEvent( event );
2618 }
2619
2620 void wxTreeListMainWindow::ExpandAll(const wxTreeItemId& item)
2621 {
2622 Expand(item);
2623 if ( IsExpanded(item) )
2624 {
2625 #if !wxCHECK_VERSION(2, 5, 0)
2626 long cookie;
2627 #else
2628 wxTreeItemIdValue cookie;
2629 #endif
2630 wxTreeItemId child = GetFirstChild(item, cookie);
2631 while ( child.IsOk() )
2632 {
2633 ExpandAll(child);
2634
2635 child = GetNextChild(item, cookie);
2636 }
2637 }
2638 }
2639
2640 void wxTreeListMainWindow::Collapse(const wxTreeItemId& itemId)
2641 {
2642 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
2643
2644 if ( !item->IsExpanded() )
2645 return;
2646
2647 wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_COLLAPSING, m_owner->GetId() );
2648 event.SetItem( (long) item );
2649 event.SetEventObject( /*this*/m_owner );
2650 if ( m_owner->ProcessEvent( event ) && !event.IsAllowed() )
2651 {
2652 // cancelled by program
2653 return;
2654 }
2655
2656 item->Collapse();
2657
2658 #if 0 // TODO why should items be collapsed recursively?
2659 wxArrayTreeListItems& children = item->GetChildren();
2660 size_t count = children.Count();
2661 for ( size_t n = 0; n < count; n++ )
2662 {
2663 Collapse(children[n]);
2664 }
2665 #endif
2666
2667 CalculatePositions();
2668
2669 RefreshSubtree(item);
2670
2671 event.SetEventType(wxEVT_COMMAND_TREE_ITEM_COLLAPSED);
2672 ProcessEvent( event );
2673 }
2674
2675 void wxTreeListMainWindow::CollapseAndReset(const wxTreeItemId& item)
2676 {
2677 Collapse(item);
2678 DeleteChildren(item);
2679 }
2680
2681 void wxTreeListMainWindow::Toggle(const wxTreeItemId& itemId)
2682 {
2683 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
2684
2685 if (item->IsExpanded())
2686 Collapse(itemId);
2687 else
2688 Expand(itemId);
2689 }
2690
2691 void wxTreeListMainWindow::Unselect()
2692 {
2693 if (m_current)
2694 {
2695 m_current->SetHilight( FALSE );
2696 RefreshLine( m_current );
2697 }
2698 }
2699
2700 void wxTreeListMainWindow::UnselectAllChildren(wxTreeListItem *item)
2701 {
2702 if (item->IsSelected())
2703 {
2704 item->SetHilight(FALSE);
2705 RefreshLine(item);
2706 }
2707
2708 if (item->HasChildren())
2709 {
2710 wxArrayTreeListItems& children = item->GetChildren();
2711 size_t count = children.Count();
2712 for ( size_t n = 0; n < count; ++n )
2713 {
2714 UnselectAllChildren(children[n]);
2715 }
2716 }
2717 }
2718
2719 void wxTreeListMainWindow::UnselectAll()
2720 {
2721 UnselectAllChildren((wxTreeListItem*)GetRootItem().m_pItem);
2722 }
2723
2724 // Recursive function !
2725 // To stop we must have crt_item<last_item
2726 // Algorithm :
2727 // Tag all next children, when no more children,
2728 // Move to parent (not to tag)
2729 // Keep going... if we found last_item, we stop.
2730 bool wxTreeListMainWindow::TagNextChildren(wxTreeListItem *crt_item, wxTreeListItem *last_item, bool select)
2731 {
2732 wxTreeListItem *parent = crt_item->GetItemParent();
2733
2734 if (parent == NULL) // This is root item
2735 return TagAllChildrenUntilLast(crt_item, last_item, select);
2736
2737 wxArrayTreeListItems& children = parent->GetChildren();
2738 int index = children.Index(crt_item);
2739 wxASSERT( index != wxNOT_FOUND ); // I'm not a child of my parent?
2740
2741 size_t count = children.Count();
2742 for (size_t n=(size_t)(index+1); n<count; ++n)
2743 {
2744 if (TagAllChildrenUntilLast(children[n], last_item, select)) return TRUE;
2745 }
2746
2747 return TagNextChildren(parent, last_item, select);
2748 }
2749
2750 bool wxTreeListMainWindow::TagAllChildrenUntilLast(wxTreeListItem *crt_item, wxTreeListItem *last_item, bool select)
2751 {
2752 crt_item->SetHilight(select);
2753 RefreshLine(crt_item);
2754
2755 if (crt_item==last_item)
2756 return TRUE;
2757
2758 if (crt_item->HasChildren())
2759 {
2760 wxArrayTreeListItems& children = crt_item->GetChildren();
2761 size_t count = children.Count();
2762 for ( size_t n = 0; n < count; ++n )
2763 {
2764 if (TagAllChildrenUntilLast(children[n], last_item, select))
2765 return TRUE;
2766 }
2767 }
2768
2769 return FALSE;
2770 }
2771
2772 void wxTreeListMainWindow::SelectItemRange(wxTreeListItem *item1, wxTreeListItem *item2)
2773 {
2774 // item2 is not necessary after item1
2775 wxTreeListItem *first=NULL, *last=NULL;
2776
2777 // choice first' and 'last' between item1 and item2
2778 if (item1->GetY()<item2->GetY())
2779 {
2780 first=item1;
2781 last=item2;
2782 }
2783 else
2784 {
2785 first=item2;
2786 last=item1;
2787 }
2788
2789 bool select = m_current->IsSelected();
2790
2791 if ( TagAllChildrenUntilLast(first,last,select) )
2792 return;
2793
2794 TagNextChildren(first,last,select);
2795 }
2796
2797 void wxTreeListMainWindow::SelectItem(const wxTreeItemId& itemId,
2798 bool unselect_others,
2799 bool extended_select)
2800 {
2801 wxCHECK_RET( itemId.IsOk(), wxT("invalid tree item") );
2802
2803 bool is_single=!(GetWindowStyleFlag() & wxTR_MULTIPLE);
2804 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
2805
2806 //wxCHECK_RET( ( (!unselect_others) && is_single),
2807 // wxT("this is a single selection tree") );
2808
2809 // to keep going anyhow !!!
2810 if (is_single)
2811 {
2812 if (item->IsSelected())
2813 return; // nothing to do
2814 unselect_others = TRUE;
2815 extended_select = FALSE;
2816 }
2817 else if ( unselect_others && item->IsSelected() )
2818 {
2819 // selection change if there is more than one item currently selected
2820 wxArrayTreeItemIds selected_items;
2821 if ( GetSelections(selected_items) == 1 )
2822 return;
2823 }
2824
2825 wxTreeEvent event( wxEVT_COMMAND_TREE_SEL_CHANGING, m_owner->GetId() );
2826 event.SetItem( (long) item );
2827 event.SetOldItem( (long) m_current );
2828 event.SetEventObject( /*this*/m_owner );
2829 // TODO : Here we don't send any selection mode yet !
2830
2831 if(m_owner->GetEventHandler()->ProcessEvent( event ) && !event.IsAllowed())
2832 return;
2833
2834 wxTreeItemId parent = GetItemParent( itemId );
2835 while (parent.IsOk())
2836 {
2837 if (!IsExpanded(parent))
2838 Expand( parent );
2839
2840 parent = GetItemParent( parent );
2841 }
2842
2843 EnsureVisible( itemId );
2844
2845 // ctrl press
2846 if (unselect_others)
2847 {
2848 if (is_single) Unselect(); // to speed up thing
2849 else UnselectAll();
2850 }
2851
2852 // shift press
2853 if (extended_select)
2854 {
2855 if ( !m_current )
2856 {
2857 m_current = m_key_current = (wxTreeListItem*)GetRootItem().m_pItem;
2858 }
2859
2860 // don't change the mark (m_current)
2861 SelectItemRange(m_current, item);
2862 }
2863 else
2864 {
2865 bool select=TRUE; // the default
2866
2867 // Check if we need to toggle hilight (ctrl mode)
2868 if (!unselect_others)
2869 select=!item->IsSelected();
2870
2871 m_current = m_key_current = item;
2872 m_current->SetHilight(select);
2873 RefreshLine( m_current );
2874 }
2875
2876 event.SetEventType(wxEVT_COMMAND_TREE_SEL_CHANGED);
2877 GetEventHandler()->ProcessEvent( event );
2878 }
2879
2880 void wxTreeListMainWindow::SelectAll(bool extended_select)
2881 {
2882 wxCHECK_RET( GetWindowStyleFlag() & wxTR_MULTIPLE, wxT("invalid tree style") );
2883
2884 wxTreeEvent event( wxEVT_COMMAND_TREE_SEL_CHANGING, m_owner->GetId() );
2885 event.SetItem( GetRootItem() );
2886 event.SetOldItem( (long) m_current );
2887 event.SetEventObject( /*this*/m_owner );
2888 // TODO : Here we don't send any selection mode yet !
2889
2890 if(m_owner->GetEventHandler()->ProcessEvent( event ) && !event.IsAllowed())
2891 return;
2892
2893 // shift press
2894 if (!extended_select)
2895 {
2896
2897 }
2898 else
2899 {
2900
2901 }
2902 #if !wxCHECK_VERSION(2, 5, 0)
2903 long cookie = 0;
2904 #else
2905 wxTreeItemIdValue cookie = 0;
2906 #endif
2907 wxTreeItemId root = GetRootItem();
2908 wxTreeListItem *first = (wxTreeListItem *)GetFirstChild (root, cookie).m_pItem;
2909 wxTreeListItem *last = (wxTreeListItem *)GetLastChild (GetRootItem()).m_pItem;
2910 if (TagAllChildrenUntilLast (first, last, true)) return;
2911 TagNextChildren (first, last, true);
2912
2913 event.SetEventType(wxEVT_COMMAND_TREE_SEL_CHANGED);
2914 GetEventHandler()->ProcessEvent( event );
2915 }
2916
2917 void wxTreeListMainWindow::FillArray(wxTreeListItem *item,
2918 wxArrayTreeItemIds &array) const
2919 {
2920 if ( item->IsSelected() )
2921 array.Add(wxTreeItemId(item));
2922
2923 if ( item->HasChildren() )
2924 {
2925 wxArrayTreeListItems& children = item->GetChildren();
2926 size_t count = children.GetCount();
2927 for ( size_t n = 0; n < count; ++n )
2928 FillArray(children[n], array);
2929 }
2930 }
2931
2932 size_t wxTreeListMainWindow::GetSelections(wxArrayTreeItemIds &array) const
2933 {
2934 array.Empty();
2935 wxTreeItemId idRoot = GetRootItem();
2936 if ( idRoot.IsOk() )
2937 {
2938 FillArray((wxTreeListItem*) idRoot.m_pItem, array);
2939 }
2940 //else: the tree is empty, so no selections
2941
2942 return array.Count();
2943 }
2944
2945 void wxTreeListMainWindow::EnsureVisible(const wxTreeItemId& item)
2946 {
2947 if (!item.IsOk()) return;
2948
2949 wxTreeListItem *gitem = (wxTreeListItem*) item.m_pItem;
2950
2951 // first expand all parent branches
2952 wxTreeListItem *parent = gitem->GetItemParent();
2953 while ( parent )
2954 {
2955 Expand(parent);
2956 parent = parent->GetItemParent();
2957 }
2958
2959 //if (parent) CalculatePositions();
2960
2961 ScrollTo(item);
2962 }
2963
2964 void wxTreeListMainWindow::ScrollTo(const wxTreeItemId &item)
2965 {
2966 if (!item.IsOk()) return;
2967
2968 // We have to call this here because the label in
2969 // question might just have been added and no screen
2970 // update taken place.
2971 if (m_dirty) wxYieldIfNeeded();
2972
2973 wxTreeListItem *gitem = (wxTreeListItem*) item.m_pItem;
2974
2975 // now scroll to the item
2976 int item_y = gitem->GetY();
2977
2978 int start_x = 0;
2979 int start_y = 0;
2980 GetViewStart( &start_x, &start_y );
2981 start_y *= PIXELS_PER_UNIT;
2982
2983 int client_h = 0;
2984 int client_w = 0;
2985 GetClientSize( &client_w, &client_h );
2986
2987 if (item_y < start_y+3)
2988 {
2989 // going down
2990 int x = 0;
2991 int y = 0;
2992 m_anchor->GetSize( x, y, this );
2993 x = m_owner->GetHeaderWindow()->GetWidth(); //m_total_col_width; // ALB
2994 y += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
2995 //x += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
2996 int x_pos = GetScrollPos( wxHORIZONTAL );
2997 // Item should appear at top
2998 SetScrollbars( PIXELS_PER_UNIT, PIXELS_PER_UNIT, x/PIXELS_PER_UNIT, y/PIXELS_PER_UNIT, x_pos, item_y/PIXELS_PER_UNIT );
2999 }
3000 else if (item_y+GetLineHeight(gitem) > start_y+client_h)
3001 {
3002 // going up
3003 int x = 0;
3004 int y = 0;
3005 m_anchor->GetSize( x, y, this );
3006 y += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
3007 //x += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
3008 x = m_owner->GetHeaderWindow()->GetWidth(); //m_total_col_width; // ALB
3009 item_y += PIXELS_PER_UNIT+2;
3010 int x_pos = GetScrollPos( wxHORIZONTAL );
3011 // Item should appear at bottom
3012 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 );
3013 }
3014 }
3015
3016 // FIXME: tree sorting functions are not reentrant and not MT-safe!
3017 static wxTreeListMainWindow *s_treeBeingSorted = NULL;
3018
3019 static int LINKAGEMODE tree_ctrl_compare_func(wxTreeListItem **item1,
3020 wxTreeListItem **item2)
3021 {
3022 wxCHECK_MSG( s_treeBeingSorted, 0, wxT("bug in wxTreeListMainWindow::SortChildren()") );
3023
3024 return s_treeBeingSorted->OnCompareItems(*item1, *item2);
3025 }
3026
3027 int wxTreeListMainWindow::OnCompareItems(const wxTreeItemId& item1,
3028 const wxTreeItemId& item2)
3029 {
3030 // ALB: delegate to m_owner, to let the user overrride the comparison
3031 //return wxStrcmp(GetItemText(item1), GetItemText(item2));
3032 return m_owner->OnCompareItems(item1, item2);
3033 }
3034
3035 void wxTreeListMainWindow::SortChildren(const wxTreeItemId& itemId)
3036 {
3037 wxCHECK_RET( itemId.IsOk(), wxT("invalid tree item") );
3038
3039 wxTreeListItem *item = (wxTreeListItem*) itemId.m_pItem;
3040
3041 wxCHECK_RET( !s_treeBeingSorted,
3042 wxT("wxTreeListMainWindow::SortChildren is not reentrant") );
3043
3044 wxArrayTreeListItems& children = item->GetChildren();
3045 if ( children.Count() > 1 )
3046 {
3047 m_dirty = TRUE;
3048
3049 s_treeBeingSorted = this;
3050 children.Sort(tree_ctrl_compare_func);
3051 s_treeBeingSorted = NULL;
3052 }
3053 //else: don't make the tree dirty as nothing changed
3054 }
3055
3056 wxTreeItemId wxTreeListMainWindow::FindItem (const wxTreeItemId& item, const wxString& str, int flags) {
3057 #if !wxCHECK_VERSION(2, 5, 0)
3058 long cookie = 0;
3059 #else
3060 wxTreeItemIdValue cookie = 0;
3061 #endif
3062 wxTreeItemId next = item;
3063 if (!next.IsOk()) next = GetSelection();
3064 if (!next.IsOk()) {
3065 if (HasFlag(wxTR_HIDE_ROOT)) {
3066 next = (wxTreeListItem*)GetFirstChild (GetRootItem().m_pItem, cookie).m_pItem;
3067 } else {
3068 next = (wxTreeListItem*)GetRootItem().m_pItem;
3069 }
3070 }
3071 if (!next.IsOk()) return item;
3072
3073 // start checking the next items
3074 wxString itemText;
3075 while (next.IsOk()) {
3076 itemText = GetItemText (next);
3077 if (flags & wxTL_SEARCH_LEVEL) {
3078 next = GetNextSibling (next);
3079 }else if (flags & wxTL_SEARCH_FULL) {
3080 wxTreeItemId n = GetFirstChild (next, cookie);
3081 if (!n.IsOk())
3082 n = GetNextSibling (next);
3083 if (!n.IsOk())
3084 n = GetNextSibling (GetItemParent (next));
3085 next = n;
3086 }else{ // wxTL_SEARCH_VISIBLE
3087 next = GetNextVisible (next);
3088 }
3089 if (!next.IsOk()) break; // done
3090 if (flags & wxTL_SEARCH_PARTIAL) {
3091 itemText = GetItemText (next).Mid (0, str.Length());
3092 }else{
3093 itemText = GetItemText (next);
3094 }
3095 if (flags & wxTL_SEARCH_NOCASE) {
3096 if (itemText.CmpNoCase (str) == 0) return next;
3097 }else{
3098 if (itemText.Cmp (str) == 0) return next;
3099 }
3100 }
3101 return item;
3102 }
3103
3104 wxImageList *wxTreeListMainWindow::GetImageList() const
3105 {
3106 return m_imageListNormal;
3107 }
3108
3109 wxImageList *wxTreeListMainWindow::GetButtonsImageList() const
3110 {
3111 return m_imageListButtons;
3112 }
3113
3114 wxImageList *wxTreeListMainWindow::GetStateImageList() const
3115 {
3116 return m_imageListState;
3117 }
3118
3119 void wxTreeListMainWindow::CalculateLineHeight()
3120 {
3121 wxClientDC dc(this);
3122 dc.SetFont( m_normalFont );
3123 m_lineHeight = (int)(dc.GetCharHeight() + m_linespacing);
3124
3125 if ( m_imageListNormal )
3126 {
3127 // Calculate a m_lineHeight value from the normal Image sizes.
3128 // May be toggle off. Then wxTreeListMainWindow will spread when
3129 // necessary (which might look ugly).
3130 int n = m_imageListNormal->GetImageCount();
3131 for (int i = 0; i < n ; i++)
3132 {
3133 int width = 0, height = 0;
3134 m_imageListNormal->GetSize(i, width, height);
3135 if (height > m_lineHeight) m_lineHeight = height + m_linespacing;
3136 }
3137 }
3138
3139 if (m_imageListButtons)
3140 {
3141 // Calculate a m_lineHeight value from the Button image sizes.
3142 // May be toggle off. Then wxTreeListMainWindow will spread when
3143 // necessary (which might look ugly).
3144 int n = m_imageListButtons->GetImageCount();
3145 for (int i = 0; i < n ; i++)
3146 {
3147 int width = 0, height = 0;
3148 m_imageListButtons->GetSize(i, width, height);
3149 if (height > m_lineHeight) m_lineHeight = height + m_linespacing;
3150 }
3151 }
3152
3153 /*? FIXME: Don't get what this code is for... Adding a line space is already done!!!
3154 if (m_lineHeight < 30)
3155 m_lineHeight += 2; // at least 2 pixels
3156 else
3157 m_lineHeight += m_lineHeight/10; // otherwise 10% extra spacing
3158 ?*/
3159 }
3160
3161 void wxTreeListMainWindow::SetImageList(wxImageList *imageList)
3162 {
3163 if (m_ownsImageListNormal) delete m_imageListNormal;
3164 m_imageListNormal = imageList;
3165 m_ownsImageListNormal = FALSE;
3166 m_dirty = TRUE;
3167 CalculateLineHeight();
3168 }
3169
3170 void wxTreeListMainWindow::SetStateImageList(wxImageList *imageList)
3171 {
3172 if (m_ownsImageListState) delete m_imageListState;
3173 m_imageListState = imageList;
3174 m_ownsImageListState = FALSE;
3175 }
3176
3177 void wxTreeListMainWindow::SetButtonsImageList(wxImageList *imageList)
3178 {
3179 if (m_ownsImageListButtons) delete m_imageListButtons;
3180 m_imageListButtons = imageList;
3181 m_ownsImageListButtons = FALSE;
3182 m_dirty = TRUE;
3183 CalculateLineHeight();
3184 }
3185
3186 void wxTreeListMainWindow::AssignImageList(wxImageList *imageList)
3187 {
3188 SetImageList(imageList);
3189 m_ownsImageListNormal = TRUE;
3190 }
3191
3192 void wxTreeListMainWindow::AssignStateImageList(wxImageList *imageList)
3193 {
3194 SetStateImageList(imageList);
3195 m_ownsImageListState = TRUE;
3196 }
3197
3198 void wxTreeListMainWindow::AssignButtonsImageList(wxImageList *imageList)
3199 {
3200 SetButtonsImageList(imageList);
3201 m_ownsImageListButtons = TRUE;
3202 }
3203
3204 // ----------------------------------------------------------------------------
3205 // helpers
3206 // ----------------------------------------------------------------------------
3207
3208 void wxTreeListMainWindow::AdjustMyScrollbars()
3209 {
3210 if (m_anchor)
3211 {
3212 int x = 0, y = 0;
3213 m_anchor->GetSize( x, y, this );
3214 y += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
3215 //x += PIXELS_PER_UNIT+2; // one more scrollbar unit + 2 pixels
3216 int x_pos = GetScrollPos( wxHORIZONTAL );
3217 int y_pos = GetScrollPos( wxVERTICAL );
3218 x = m_owner->GetHeaderWindow()->GetWidth() + 2;
3219 if(x < GetClientSize().GetWidth()) x_pos = 0;
3220 //m_total_col_width + 2; // ALB
3221 SetScrollbars( PIXELS_PER_UNIT, PIXELS_PER_UNIT, x/PIXELS_PER_UNIT,
3222 y/PIXELS_PER_UNIT, x_pos, y_pos );
3223 }
3224 else
3225 {
3226 SetScrollbars( 0, 0, 0, 0 );
3227 }
3228 }
3229
3230 int wxTreeListMainWindow::GetLineHeight(wxTreeListItem *item) const
3231 {
3232 if (GetWindowStyleFlag() & wxTR_HAS_VARIABLE_ROW_HEIGHT)
3233 return item->GetHeight();
3234 else
3235 return m_lineHeight;
3236 }
3237
3238 void wxTreeListMainWindow::PaintItem(wxTreeListItem *item, wxDC& dc)
3239 {
3240 wxTreeItemAttr *attr = item->GetAttributes();
3241 if (attr && attr->HasFont()) {
3242 dc.SetFont(attr->GetFont());
3243 }else if (item->IsBold()) {
3244 dc.SetFont(m_boldFont);
3245 }
3246 wxColour colText;
3247 if (attr && attr->HasTextColour()) {
3248 colText = attr->GetTextColour();
3249 }else{
3250 colText = GetForegroundColour();
3251 }
3252
3253 wxPen *pen =
3254 #ifndef __WXMAC__
3255 // don't draw rect outline if we already have the
3256 // background color under Mac
3257 (item->IsSelected() && m_hasFocus) ? wxBLACK_PEN :
3258 #endif // !__WXMAC__
3259 wxTRANSPARENT_PEN;
3260
3261 long text_w = 0, text_h = 0;
3262
3263 dc.GetTextExtent( item->GetText(GetMainColumn()), &text_w, &text_h );
3264
3265 int total_h = GetLineHeight(item);
3266
3267 if (item->IsSelected() && HasFlag (wxTR_FULL_ROW_HIGHLIGHT)) {
3268 dc.SetBrush(*(m_hasFocus ? m_hilightBrush : m_hilightUnfocusedBrush));
3269 int offset = HasFlag(wxTR_ROW_LINES) ? 1 : 0;
3270 dc.DrawRectangle (0, item->GetY() + offset,
3271 m_owner->GetHeaderWindow()->GetWidth(), total_h-offset);
3272 colText = wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHTTEXT);
3273 }else{
3274 wxColour colBg;
3275 if (attr && attr->HasBackgroundColour()) {
3276 colBg = attr->GetBackgroundColour();
3277 }else{
3278 colBg = m_backgroundColour;
3279 }
3280 dc.SetBrush(wxBrush(colBg, wxTRANSPARENT));
3281 }
3282
3283 dc.SetBackgroundMode(wxTRANSPARENT);
3284 int text_extraH = (total_h > text_h) ? (total_h - text_h)/2 : 0;
3285 int img_extraH = (total_h > m_imgHeight)? (total_h-m_imgHeight)/2: 0;
3286 int x_colstart = 0;
3287 for ( size_t i = 0; i < GetColumnCount(); ++i ) {
3288 if (!m_owner->GetHeaderWindow()->GetColumnShown(i)) continue;
3289 int colwidth = m_owner->GetHeaderWindow()->GetColumnWidth(i);
3290 int image;
3291 int image_x = 0;
3292 int image_w = 0;
3293 if(i == GetMainColumn()) {
3294 image = item->GetCurrentImage();
3295 if (HasButtons()) {
3296 image_x = item->GetX() + (m_btnWidth-m_btnWidth2) + LINEATROOT;
3297 }else{
3298 image_x = item->GetX() - m_imgWidth2;
3299 }
3300 }
3301 else
3302 {
3303 image = item->GetImage(i);
3304 image_x = x_colstart + MARGIN;
3305 }
3306 if (image != NO_IMAGE) image_w = m_imgWidth + MARGIN;
3307
3308 // honor text alignment
3309 wxString text = item->GetText(i);
3310 switch ( m_owner->GetHeaderWindow()->GetColumn(i).GetAlignment() ) {
3311 case wxTL_ALIGN_LEFT:
3312 // already left aligned
3313 break;
3314 case wxTL_ALIGN_RIGHT:
3315 dc.GetTextExtent(text, &text_w, NULL);
3316 image_x = x_colstart + colwidth - (image_w + text_w + MARGIN);
3317 break;
3318 case wxTL_ALIGN_CENTER:
3319 dc.GetTextExtent(text, &text_w, NULL);
3320 int w = colwidth - image_w - text_w;
3321 image_x = x_colstart + (w > 0)? w: 0;
3322 break;
3323 }
3324 int text_x = image_x + image_w;
3325
3326 if (item->IsSelected() && (i==GetMainColumn()) && !HasFlag (wxTR_FULL_ROW_HIGHLIGHT))
3327 {
3328 dc.SetBrush(*(m_hasFocus ? m_hilightBrush : m_hilightUnfocusedBrush));
3329 int offset = HasFlag (wxTR_ROW_LINES) ? 1 : 0;
3330 dc.DrawRectangle (text_x, item->GetY() + offset, text_w, total_h-offset);
3331 dc.SetBackgroundMode(wxTRANSPARENT);
3332 dc.SetTextForeground(wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHTTEXT));
3333 }else{
3334 dc.SetTextForeground(colText);
3335 }
3336
3337 dc.SetPen(*pen);
3338
3339 wxDCClipper clipper (dc, x_colstart, item->GetY(), colwidth, total_h);
3340 if (image != NO_IMAGE)
3341 {
3342 int image_y = item->GetY() + img_extraH;
3343 m_imageListNormal->Draw ( image, dc, image_x, image_y,
3344 wxIMAGELIST_DRAW_TRANSPARENT );
3345 }
3346 int text_y = item->GetY() + text_extraH;
3347 dc.DrawText ( text, (wxCoord)text_x, (wxCoord)text_y );
3348
3349 x_colstart += colwidth;
3350 }
3351
3352 // restore normal font
3353 dc.SetFont( m_normalFont );
3354 }
3355
3356 // Now y stands for the top of the item, whereas it used to stand for middle !
3357 void wxTreeListMainWindow::PaintLevel (wxTreeListItem *item, wxDC &dc,
3358 int level, int &y, int x_colstart )
3359 {
3360 // Handle hide root (only level 0)
3361 if (HasFlag(wxTR_HIDE_ROOT) && (level == 0)) {
3362 // always expand hidden root
3363 wxArrayTreeListItems& children = item->GetChildren();
3364 int n;
3365 for (n = 0; n < (int)children.Count(); n++) {
3366 PaintLevel (children[n], dc, 1, y, x_colstart);
3367 }
3368 // end after expanding root
3369 return;
3370 }
3371
3372 // calculate position of vertical lines
3373 int x = x_colstart + MARGIN; // start of column
3374 if (HasFlag (wxTR_LINES_AT_ROOT)) x += LINEATROOT; // space for lines at root
3375 if (HasButtons()) {
3376 x += m_btnWidth2; // middle of button
3377 }else{
3378 if (m_imgWidth > 0) x += m_imgWidth2; // middle of image
3379 }
3380 if (!HasFlag (wxTR_HIDE_ROOT)) {
3381 x += m_indent * level; // indent according to level
3382 }else{
3383 if (level > 0) x += m_indent * (level-1); // but not level 1
3384 }
3385
3386 // handle column text
3387 item->SetX (x);
3388 item->SetY (y);
3389
3390 int h = GetLineHeight(item);
3391 int y_top = y;
3392 int y_mid = y_top + (h/2);
3393 y += h;
3394
3395 int exposed_x = dc.LogicalToDeviceX(0);
3396 int exposed_y = dc.LogicalToDeviceY(y_top);
3397
3398 if (IsExposed(exposed_x, exposed_y, 10000, h)) // 10000 = very much
3399 {
3400 if (HasFlag(wxTR_ROW_LINES))
3401 {
3402 //dc.DestroyClippingRegion();
3403 int total_width = m_owner->GetHeaderWindow()->GetWidth();
3404 // if the background colour is white, choose a
3405 // contrasting color for the lines
3406 dc.SetPen (*((GetBackgroundColour() == *wxWHITE)?
3407 wxMEDIUM_GREY_PEN : wxWHITE_PEN));
3408 dc.DrawLine(0, y_top, total_width, y_top);
3409 dc.DrawLine(0, y, total_width, y);
3410 }
3411
3412 // restore DC objects
3413 dc.SetBrush(*wxWHITE_BRUSH);
3414 dc.SetPen(m_dottedPen);
3415
3416 if (((level == 0) || ((level == 1) && HasFlag(wxTR_HIDE_ROOT))) &&
3417 HasFlag(wxTR_LINES_AT_ROOT) && !HasFlag(wxTR_NO_LINES)) {
3418 int rootPos = x_colstart + MARGIN;
3419 dc.DrawLine (rootPos, y_mid, rootPos+LINEATROOT, y_mid);
3420 }
3421
3422 size_t clip_width = m_owner->GetHeaderWindow()->
3423 GetColumn(m_main_column).GetWidth();
3424
3425 if (item->HasPlus() && HasButtons()) // should the item show a button?
3426 {
3427 // clip to the column width
3428 wxDCClipper clipper(dc, x_colstart, y_top, clip_width, 10000);
3429
3430 if (m_imageListButtons != NULL)
3431 {
3432 // draw the image button here
3433 int image = wxTreeItemIcon_Normal;
3434 if (item->IsExpanded()) image = wxTreeItemIcon_Expanded;
3435 if (item->IsSelected())
3436 image += wxTreeItemIcon_Selected - wxTreeItemIcon_Normal;
3437 int xx = x + m_btnWidth2;
3438 int yy = y_mid - m_btnHeight2;
3439 dc.SetClippingRegion(xx, yy, m_btnWidth, m_btnHeight);
3440 m_imageListButtons->Draw(image, dc, xx, yy,
3441 wxIMAGELIST_DRAW_TRANSPARENT);
3442 dc.DestroyClippingRegion();
3443 }
3444 else if (HasFlag(wxTR_TWIST_BUTTONS))
3445 {
3446 // draw the twisty button here
3447 dc.SetPen(*wxBLACK_PEN);
3448 dc.SetBrush(*m_hilightBrush);
3449
3450 wxPoint button[3];
3451
3452 if (item->IsExpanded())
3453 {
3454 button[0].x = x - (m_btnWidth2+1);
3455 button[0].y = y_mid - (m_btnHeight/3);
3456 button[1].x = x + (m_btnWidth2+1);
3457 button[1].y = button[0].y;
3458 button[2].x = x;
3459 button[2].y = button[0].y + (m_btnHeight2+1);
3460 }
3461 else
3462 {
3463 button[0].x = x - (m_btnWidth/3);
3464 button[0].y = y_mid - (m_btnHeight2+1);
3465 button[1].x = button[0].x;
3466 button[1].y = y_mid + (m_btnHeight2+1);
3467 button[2].x = button[0].x + (m_btnWidth2+1);
3468 button[2].y = y_mid;
3469 }
3470 dc.DrawPolygon(3, button);
3471
3472 dc.SetPen(m_dottedPen);
3473 }
3474 else // if (HasFlag(wxTR_HAS_BUTTONS))
3475 {
3476 // draw the plus sign here
3477 dc.SetPen(*wxGREY_PEN);
3478 dc.SetBrush(*wxWHITE_BRUSH);
3479 dc.DrawRectangle (x-m_btnWidth2, y_mid-m_btnHeight2,
3480 m_btnWidth, m_btnHeight);
3481 dc.SetPen(*wxBLACK_PEN);
3482 dc.DrawLine (x-(m_btnWidth2-3), y_mid,
3483 x+(m_btnWidth2-2), y_mid);
3484 if (!item->IsExpanded())
3485 dc.DrawLine (x, y_mid-(m_btnHeight2-2),
3486 x, y_mid+(m_btnHeight2-1));
3487 dc.SetPen(m_dottedPen);
3488 }
3489
3490 if (!HasFlag(wxTR_NO_LINES)) {
3491 if (!(level == 0) && !((level == 1) && HasFlag(wxTR_HIDE_ROOT))) {
3492 if (m_imgWidth > 0) {
3493 dc.DrawLine(x+m_btnWidth2, y_mid, x+m_indent-m_imgWidth2, y_mid);
3494 }else{
3495 dc.DrawLine(x+m_btnWidth2, y_mid, x+m_btnWidth2+LINEATROOT-MARGIN, y_mid);
3496 }
3497 }
3498 }
3499 }
3500 else if (!HasFlag(wxTR_NO_LINES)) // no button; maybe a line?
3501 {
3502 // clip to the column width
3503 wxDCClipper clipper(dc, x_colstart, y_top, clip_width, 10000);
3504
3505 // draw the horizontal line here
3506 if (!(level == 0) && !((level == 1) && HasFlag(wxTR_HIDE_ROOT))) {
3507 int x2 = x;
3508 if (!HasButtons()) x2 = x - m_indent;
3509 if (m_imgWidth > 0) {
3510 dc.DrawLine(x2, y_mid, x2+m_indent-m_imgWidth2, y_mid);
3511 }else{
3512 dc.DrawLine(x2, y_mid, x2+m_btnWidth2+LINEATROOT-MARGIN, y_mid);
3513 }
3514 }
3515 }
3516
3517 // draw item
3518 PaintItem(item, dc);
3519 }
3520
3521 // restore DC objects
3522 dc.SetBrush(*wxWHITE_BRUSH);
3523 dc.SetPen(m_dottedPen);
3524 dc.SetTextForeground(*wxBLACK);
3525
3526 if (item->IsExpanded())
3527 {
3528 wxArrayTreeListItems& children = item->GetChildren();
3529
3530 // clip to the column width
3531 size_t clip_width = m_owner->GetHeaderWindow()->
3532 GetColumn(m_main_column).GetWidth();
3533 wxDCClipper clipper(dc, x_colstart, y_top, clip_width, 10000);
3534
3535 // process lower levels
3536 int oldY;
3537 if (m_imgWidth > 0) {
3538 oldY = y_mid + m_imgHeight2;
3539 }else{
3540 oldY = y_mid + h/2;
3541 }
3542 int y2;
3543 int n;
3544 for (n = 0; n < (int)children.Count(); ++n) {
3545
3546 if (!HasFlag(wxTR_NO_LINES))
3547 {
3548 // draw line down to last child
3549 if (children[n]->HasPlus() && HasButtons()) {
3550 y2 = y + h/2 - m_btnHeight2;
3551 if (HasButtons()) {
3552 dc.DrawLine(x+m_indent, oldY, x+m_indent, y2);
3553 }else{
3554 dc.DrawLine(x, oldY, x, y2);
3555 }
3556 oldY = y2 + m_btnHeight;
3557 }else{
3558 y2 = y + h/2;
3559 if (HasButtons()) {
3560 dc.DrawLine(x+m_indent, oldY, x+m_indent, y2);
3561 }else{
3562 dc.DrawLine(x, oldY, x, y2);
3563 }
3564 oldY = y2;
3565 }
3566 }
3567
3568 PaintLevel (children[n], dc, level+1, y, x_colstart);
3569 }
3570 }
3571 }
3572
3573 void wxTreeListMainWindow::DrawDropEffect(wxTreeListItem *item)
3574 {
3575 if ( item )
3576 {
3577 if ( item->HasPlus() )
3578 {
3579 // it's a folder, indicate it by a border
3580 DrawBorder(item);
3581 }
3582 else
3583 {
3584 // draw a line under the drop target because the item will be
3585 // dropped there
3586 DrawLine(item, TRUE /* below */);
3587 }
3588
3589 SetCursor(wxCURSOR_BULLSEYE);
3590 }
3591 else
3592 {
3593 // can't drop here
3594 SetCursor(wxCURSOR_NO_ENTRY);
3595 }
3596 }
3597
3598 void wxTreeListMainWindow::DrawBorder(const wxTreeItemId &item)
3599 {
3600 wxCHECK_RET( item.IsOk(), _T("invalid item in wxTreeListMainWindow::DrawLine") );
3601
3602 wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
3603
3604 wxClientDC dc(this);
3605 PrepareDC( dc );
3606 dc.SetLogicalFunction(wxINVERT);
3607 dc.SetBrush(*wxTRANSPARENT_BRUSH);
3608
3609 int w = i->GetWidth() + 2;
3610 int h = GetLineHeight(i) + 2;
3611
3612 dc.DrawRectangle( i->GetX() - 1, i->GetY() - 1, w, h);
3613 }
3614
3615 void wxTreeListMainWindow::DrawLine(const wxTreeItemId &item, bool below)
3616 {
3617 wxCHECK_RET( item.IsOk(), _T("invalid item in wxTreeListMainWindow::DrawLine") );
3618
3619 wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
3620
3621 wxClientDC dc(this);
3622 PrepareDC( dc );
3623 dc.SetLogicalFunction(wxINVERT);
3624
3625 int x = i->GetX(),
3626 y = i->GetY();
3627 if ( below )
3628 {
3629 y += GetLineHeight(i) - 1;
3630 }
3631
3632 dc.DrawLine( x, y, x + i->GetWidth(), y);
3633 }
3634
3635 // ----------------------------------------------------------------------------
3636 // wxWindows callbacks
3637 // ----------------------------------------------------------------------------
3638
3639 void wxTreeListMainWindow::OnPaint( wxPaintEvent &WXUNUSED(event) )
3640 {
3641 wxPaintDC dc(this);
3642
3643 PrepareDC( dc );
3644
3645 if(!GetColumnCount()) return; // ALB
3646
3647 if ( !m_anchor)
3648 return;
3649
3650 // calculate button size
3651 m_btnWidth = 0, m_btnWidth2 = 0;
3652 m_btnHeight = 0, m_btnHeight2 = 0;
3653 if (m_imageListButtons) {
3654 m_imageListButtons->GetSize (0, m_btnWidth, m_btnHeight);
3655 }else if (HasButtons()) {
3656 m_btnWidth = BTNWIDTH;
3657 m_btnHeight = BTNHEIGHT;
3658 }
3659 m_btnWidth2 = m_btnWidth/2;
3660 m_btnHeight2 = m_btnHeight/2;
3661
3662 // calculate image size
3663 m_imgWidth = 0, m_imgWidth2 = 0;
3664 m_imgHeight = 0, m_imgHeight2 = 0;
3665 if (m_imageListNormal) {
3666 m_imageListNormal->GetSize (0, m_imgWidth, m_imgHeight);
3667 m_imgWidth += 4; //? ToDo: Why + 4?
3668 }
3669 m_imgWidth2 = m_imgWidth/2;
3670 m_imgHeight2 = m_imgHeight/2;
3671
3672 // calculate indent size
3673 int btnIndent = HasButtons()? m_btnWidth + LINEATROOT: 0;
3674 m_indent = wxMax (MININDENT, wxMax (m_imgWidth, btnIndent)) + MARGIN;
3675
3676 // set default values
3677 dc.SetFont( m_normalFont );
3678 dc.SetPen( m_dottedPen );
3679
3680 // this is now done dynamically
3681 //if(GetImageList() == NULL)
3682 // m_lineHeight = (int)(dc.GetCharHeight() + 4);
3683
3684 // calculate column start and paint
3685 int x_colstart = 0;
3686 int i = 0;
3687 for (i = 0; i < (int)GetMainColumn(); ++i) {
3688 if (!m_owner->GetHeaderWindow()->GetColumnShown(i)) continue;
3689 x_colstart += m_owner->GetHeaderWindow()->GetColumnWidth (i);
3690 }
3691 int y = 0;
3692 PaintLevel ( m_anchor, dc, 0, y, x_colstart );
3693 }
3694
3695 void wxTreeListMainWindow::OnSetFocus( wxFocusEvent &event )
3696 {
3697 m_hasFocus = TRUE;
3698
3699 RefreshSelected();
3700
3701 event.Skip();
3702 }
3703
3704 void wxTreeListMainWindow::OnKillFocus( wxFocusEvent &event )
3705 {
3706 m_hasFocus = FALSE;
3707
3708 RefreshSelected();
3709
3710 event.Skip();
3711 }
3712
3713 void wxTreeListMainWindow::OnChar( wxKeyEvent &event )
3714 {
3715 wxTreeEvent te( wxEVT_COMMAND_TREE_KEY_DOWN, m_owner->GetId() );
3716 te.SetKeyEvent( event );
3717 te.SetEventObject( /*this*/m_owner );
3718 if ( m_owner->GetEventHandler()->ProcessEvent( te ) )
3719 {
3720 // intercepted by the user code
3721 return;
3722 }
3723
3724 if ( !m_current )
3725 {
3726 if (HasFlag(wxTR_HIDE_ROOT)) {
3727 #if !wxCHECK_VERSION(2, 5, 0)
3728 long cookie = 0;
3729 #else
3730 wxTreeItemIdValue cookie = 0;
3731 #endif
3732 m_current = m_key_current = (wxTreeListItem*)GetFirstChild (GetRootItem().m_pItem, cookie).m_pItem;
3733 }
3734 else
3735 {
3736 m_current = m_key_current = (wxTreeListItem*)GetRootItem().m_pItem;
3737 }
3738 }
3739
3740 // how should the selection work for this event?
3741 bool is_multiple, extended_select, unselect_others;
3742 EventFlagsToSelType(GetWindowStyleFlag(),
3743 event.ShiftDown(),
3744 event.ControlDown(),
3745 is_multiple, extended_select, unselect_others);
3746
3747 // + : Expand (not on Win32)
3748 // - : Collaspe (not on Win32)
3749 // * : Expand all/Collapse all
3750 // ' ' | return : activate
3751 // up : go up (not last children!)
3752 // down : go down
3753 // left : go to parent (or collapse on Win32)
3754 // right : open if parent and go next (or expand on Win32)
3755 // home : go to root
3756 // end : go to last item without opening parents
3757 switch (event.GetKeyCode())
3758 {
3759 #ifndef __WXMSW__ // mimic the standard win32 tree ctrl
3760 case '+':
3761 case WXK_ADD:
3762 if (m_current->HasPlus() && !IsExpanded(m_current))
3763 {
3764 Expand (m_current);
3765 }
3766 break;
3767 #endif // __WXMSW__
3768
3769 case '*':
3770 case WXK_MULTIPLY:
3771 if ( !IsExpanded(m_current) )
3772 {
3773 // expand all
3774 ExpandAll (m_current);
3775 break;
3776 }
3777 //else: fall through to Collapse() it
3778
3779 #ifndef __WXMSW__ // mimic the standard wxTreeCtrl behaviour
3780 case '-':
3781 case WXK_SUBTRACT:
3782 if (IsExpanded(m_current))
3783 {
3784 Collapse (m_current);
3785 }
3786 break;
3787 #endif // __WXMSW__
3788
3789 case ' ':
3790 case WXK_RETURN:
3791 {
3792 wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_ACTIVATED,
3793 m_owner->GetId() );
3794 event.SetItem( (long) m_current);
3795 event.SetEventObject( /*this*/m_owner );
3796 m_owner->GetEventHandler()->ProcessEvent( event );
3797 }
3798 break;
3799
3800 // backspace goes to the parent, sends "root" activation
3801 case WXK_BACK:
3802 {
3803 wxTreeItemId prev = GetItemParent( m_current );
3804 if ((prev == GetRootItem()) && HasFlag(wxTR_HIDE_ROOT))
3805 {
3806 // don't go to root if it is hidden
3807 prev = GetPrevSibling( m_current );
3808 }
3809 if (prev)
3810 {
3811 SelectItem( prev, unselect_others, extended_select );
3812 EnsureVisible( prev );
3813 }
3814 }
3815 break;
3816
3817 // up goes to the previous sibling or to the last
3818 // of its children if it's expanded
3819 case WXK_UP:
3820 {
3821 wxTreeItemId prev = GetPrevSibling( m_key_current );
3822 if (!prev)
3823 {
3824 prev = GetItemParent( m_key_current );
3825 if ((prev == GetRootItem()) && HasFlag(wxTR_HIDE_ROOT))
3826 {
3827 break; // don't go to root if it is hidden
3828 }
3829 if (prev)
3830 {
3831 #if !wxCHECK_VERSION(2, 5, 0)
3832 long cookie = 0;
3833 #else
3834 wxTreeItemIdValue cookie = 0;
3835 #endif
3836 wxTreeItemId current = m_key_current;
3837 // TODO: Huh? If we get here, we'd better be the first child of our parent. How else could it be?
3838 if (current == GetFirstChild( prev, cookie ))
3839 {
3840 // otherwise we return to where we came from
3841 SelectItem( prev, unselect_others, extended_select );
3842 m_key_current= (wxTreeListItem*) prev.m_pItem;
3843 EnsureVisible( prev );
3844 break;
3845 }
3846 }
3847 }
3848 if (prev)
3849 {
3850 while ( IsExpanded(prev) && HasChildren(prev) )
3851 {
3852 wxTreeItemId child = GetLastChild(prev);
3853 if ( !child )
3854 {
3855 break;
3856 }
3857 prev = child;
3858 }
3859
3860 SelectItem( prev, unselect_others, extended_select );
3861 m_key_current=(wxTreeListItem*) prev.m_pItem;
3862 EnsureVisible( prev );
3863 }
3864 }
3865 break;
3866
3867 // left arrow goes to the parent
3868 case WXK_LEFT:
3869 if (IsExpanded(m_current))
3870 {
3871 Collapse(m_current);
3872 }
3873 else
3874 {
3875 wxTreeItemId prev = GetItemParent( m_current );
3876 if ((prev == GetRootItem()) && HasFlag(wxTR_HIDE_ROOT))
3877 {
3878 // don't go to root if it is hidden
3879 prev = GetPrevSibling( m_current );
3880 }
3881 if (prev)
3882 {
3883 SelectItem( prev, unselect_others, extended_select );
3884 EnsureVisible( prev );
3885 }
3886 }
3887 break;
3888
3889 case WXK_RIGHT:
3890 #if defined(__WXMSW__) // mimic the standard win32 tree ctrl
3891 if (m_current->HasPlus() && !IsExpanded(m_current))
3892 {
3893 Expand(m_current);
3894 break;
3895 }
3896 #endif // __WXMSW__
3897
3898 // this works the same as the down arrow except that we
3899 // also expand the item if it wasn't expanded yet
3900 Expand(m_current);
3901 // fall through
3902
3903 case WXK_DOWN:
3904 {
3905 if (IsExpanded(m_key_current) && HasChildren(m_key_current))
3906 {
3907 #if !wxCHECK_VERSION(2, 5, 0)
3908 long cookie = 0;
3909 #else
3910 wxTreeItemIdValue cookie = 0;
3911 #endif
3912 wxTreeItemId child = GetFirstChild( m_key_current, cookie );
3913 if (child) {
3914 SelectItem( child, unselect_others, extended_select );
3915 m_key_current=(wxTreeListItem*) child.m_pItem;
3916 EnsureVisible( child );
3917 break;
3918 }
3919 }
3920 wxTreeItemId next = GetNextSibling( m_key_current );
3921 if (!next)
3922 {
3923 wxTreeItemId current = m_key_current;
3924 while (current && !next)
3925 {
3926 current = GetItemParent( current );
3927 if (current) next = GetNextSibling( current );
3928 }
3929 }
3930 if (next)
3931 {
3932 SelectItem( next, unselect_others, extended_select );
3933 m_key_current=(wxTreeListItem*) next.m_pItem;
3934 EnsureVisible( next );
3935 }
3936 }
3937 break;
3938
3939 // <End> selects the last visible tree item
3940 case WXK_END:
3941 {
3942 wxTreeItemId last = GetRootItem();
3943
3944 while ( last.IsOk() && IsExpanded(last) )
3945 {
3946 wxTreeItemId lastChild = GetLastChild(last);
3947
3948 // it may happen if the item was expanded but then all of
3949 // its children have been deleted - so IsExpanded() returned
3950 // TRUE, but GetLastChild() returned invalid item
3951 if ( !lastChild )
3952 break;
3953
3954 last = lastChild;
3955 }
3956
3957 if ( last.IsOk() )
3958 {
3959 SelectItem( last, unselect_others, extended_select );
3960 EnsureVisible( last );
3961 }
3962 }
3963 break;
3964
3965 // <Home> selects the root item
3966 case WXK_HOME:
3967 {
3968 wxTreeItemId prev = GetRootItem();
3969 if (!prev) break;
3970 if (HasFlag(wxTR_HIDE_ROOT))
3971 {
3972 #if !wxCHECK_VERSION(2, 5, 0)
3973 long cookie = 0;
3974 #else
3975 wxTreeItemIdValue cookie = 0;
3976 #endif
3977 prev = GetFirstChild(prev, cookie);
3978 if (!prev) break;
3979 }
3980 SelectItem( prev, unselect_others, extended_select );
3981 EnsureVisible( prev );
3982 }
3983 break;
3984
3985 default:
3986 if (event.m_keyCode >= (int)' ') {
3987 if (!m_findTimer->IsRunning()) m_findStr.Clear();
3988 m_findStr.Append (event.m_keyCode);
3989 m_findTimer->Start (500, wxTIMER_ONE_SHOT);
3990 wxTreeItemId dummy = (wxTreeItemId*)NULL;
3991 wxTreeItemId item = FindItem (dummy, m_findStr, wxTL_SEARCH_VISIBLE |
3992 wxTL_SEARCH_PARTIAL |
3993 wxTL_SEARCH_NOCASE);
3994 if (item.IsOk()) {
3995 EnsureVisible (item);
3996 SelectItem (item);
3997 }
3998 }
3999 event.Skip();
4000 }
4001 }
4002
4003 wxTreeItemId wxTreeListMainWindow::HitTest(const wxPoint& point, int& flags,
4004 int& column)
4005 {
4006 // JACS: removed wxYieldIfNeeded() because it can cause the window
4007 // to be deleted from under us if a close window event is pending
4008
4009 int w, h;
4010 GetSize(&w, &h);
4011 flags=0;
4012 column = -1;
4013 if (point.x<0) flags |= wxTREE_HITTEST_TOLEFT;
4014 if (point.x>w) flags |= wxTREE_HITTEST_TORIGHT;
4015 if (point.y<0) flags |= wxTREE_HITTEST_ABOVE;
4016 if (point.y>h) flags |= wxTREE_HITTEST_BELOW;
4017 if (flags) return wxTreeItemId();
4018
4019 if (m_anchor == NULL)
4020 {
4021 flags = wxTREE_HITTEST_NOWHERE;
4022 return wxTreeItemId();
4023 }
4024
4025 wxClientDC dc(this);
4026 PrepareDC(dc);
4027 wxCoord x = dc.DeviceToLogicalX( point.x );
4028 wxCoord y = dc.DeviceToLogicalY( point.y );
4029 wxTreeListItem *hit = m_anchor->HitTest(wxPoint(x, y), this, flags,
4030 column, 0);
4031 if (hit == NULL)
4032 {
4033 flags = wxTREE_HITTEST_NOWHERE;
4034 return wxTreeItemId();
4035 }
4036 return hit;
4037 }
4038
4039 // get the bounding rectangle of the item (or of its label only)
4040 bool wxTreeListMainWindow::GetBoundingRect(const wxTreeItemId& item,
4041 wxRect& rect,
4042 bool WXUNUSED(textOnly)) const
4043 {
4044 wxCHECK_MSG( item.IsOk(), FALSE, _T("invalid item in wxTreeListMainWindow::GetBoundingRect") );
4045
4046 wxTreeListItem *i = (wxTreeListItem*) item.m_pItem;
4047
4048 int startX, startY;
4049 GetViewStart(& startX, & startY);
4050
4051 rect.x = i->GetX() - startX*PIXELS_PER_UNIT;
4052 rect.y = i->GetY() - startY*PIXELS_PER_UNIT;
4053 rect.width = i->GetWidth();
4054 //rect.height = i->GetHeight();
4055 rect.height = GetLineHeight(i);
4056
4057 return TRUE;
4058 }
4059
4060 /* **** */
4061
4062 void wxTreeListMainWindow::Edit( const wxTreeItemId& item )
4063 {
4064 if (!item.IsOk()) return;
4065
4066 m_currentEdit = (wxTreeListItem*) item.m_pItem;
4067
4068 wxTreeEvent te( wxEVT_COMMAND_TREE_BEGIN_LABEL_EDIT, m_owner->GetId() );
4069 te.SetItem( (long) m_currentEdit);
4070 te.SetEventObject( /*this*/m_owner );
4071 m_owner->GetEventHandler()->ProcessEvent( te );
4072
4073 if (!te.IsAllowed()) return;
4074
4075 // We have to call this here because the label in
4076 // question might just have been added and no screen
4077 // update taken place.
4078 if (m_dirty) wxYieldIfNeeded();
4079
4080 wxString s = m_currentEdit->GetText(/*ALB*/m_main_column);
4081 int x = m_currentEdit->GetX() + m_imgWidth2;
4082 int y = m_currentEdit->GetY();
4083 int w = wxMin (m_currentEdit->GetWidth(),
4084 m_owner->GetHeaderWindow()->GetWidth()) - m_imgWidth2;
4085 int h = m_currentEdit->GetHeight() + 2;
4086 wxClientDC dc(this);
4087 PrepareDC( dc );
4088 x = dc.LogicalToDeviceX( x );
4089 y = dc.LogicalToDeviceY( y );
4090
4091 wxTreeListTextCtrl *text = new wxTreeListTextCtrl(this, -1,
4092 &m_renameAccept,
4093 &m_renameRes,
4094 this,
4095 s,
4096 wxPoint (x,y),
4097 wxSize (w,h));
4098 text->SetFocus();
4099 }
4100
4101 void wxTreeListMainWindow::OnRenameTimer()
4102 {
4103 Edit( m_current );
4104 }
4105
4106 void wxTreeListMainWindow::OnRenameAccept()
4107 {
4108 // TODO if the validator fails this causes a crash
4109 wxTreeEvent le( wxEVT_COMMAND_TREE_END_LABEL_EDIT, m_owner->GetId() );
4110 le.SetItem( (long) m_currentEdit );
4111 le.SetEventObject( /*this*/m_owner );
4112 le.SetLabel( m_renameRes );
4113 m_owner->GetEventHandler()->ProcessEvent( le );
4114
4115 if (!le.IsAllowed()) return;
4116
4117 SetItemText( m_currentEdit, m_renameRes );
4118 }
4119
4120 void wxTreeListMainWindow::OnMouse( wxMouseEvent &event )
4121 {
4122 if ( !m_anchor ) return;
4123
4124 // we process left mouse up event (enables in-place edit), right down
4125 // (pass to the user code), left dbl click (activate item) and
4126 // dragging/moving events for items drag-and-drop
4127 if ( !(event.LeftDown() ||
4128 event.LeftUp() ||
4129 event.RightDown() ||
4130 event.LeftDClick() ||
4131 event.Dragging() ||
4132 ((event.Moving() || event.RightUp()) && m_isDragging)) )
4133 {
4134 event.Skip();
4135 return;
4136 }
4137
4138 if ( event.LeftDown() )
4139 SetFocus();
4140
4141 wxClientDC dc(this);
4142 PrepareDC(dc);
4143 wxCoord x = dc.DeviceToLogicalX( event.GetX() );
4144 wxCoord y = dc.DeviceToLogicalY( event.GetY() );
4145
4146 int flags = 0;
4147 wxTreeListItem *item = m_anchor->HitTest(wxPoint(x,y), this, flags, 0);
4148
4149 if ( event.Dragging() && !m_isDragging )
4150 {
4151 if (m_dragCount == 0)
4152 m_dragStart = wxPoint(x,y);
4153
4154 m_dragCount++;
4155
4156 if (m_dragCount != 3)
4157 {
4158 // wait until user drags a bit further...
4159 return;
4160 }
4161
4162 wxEventType command = event.RightIsDown()
4163 ? wxEVT_COMMAND_TREE_BEGIN_RDRAG
4164 : wxEVT_COMMAND_TREE_BEGIN_DRAG;
4165
4166 wxTreeEvent nevent( command,/*ALB*/ m_owner->GetId() );
4167 nevent.SetItem( (long) m_current);
4168 nevent.SetEventObject(/*this*/m_owner); // ALB
4169
4170 // by default the dragging is not supported, the user code must
4171 // explicitly allow the event for it to take place
4172 nevent.Veto();
4173
4174 if ( m_owner->GetEventHandler()->ProcessEvent(nevent) &&
4175 nevent.IsAllowed() )
4176 {
4177 // we're going to drag this item
4178 m_isDragging = TRUE;
4179
4180 // remember the old cursor because we will change it while
4181 // dragging
4182 m_oldCursor = m_cursor;
4183
4184 // in a single selection control, hide the selection temporarily
4185 if ( !(GetWindowStyleFlag() & wxTR_MULTIPLE) )
4186 {
4187 m_oldSelection = (wxTreeListItem*) GetSelection().m_pItem;
4188
4189 if ( m_oldSelection )
4190 {
4191 m_oldSelection->SetHilight(FALSE);
4192 RefreshLine(m_oldSelection);
4193 }
4194 }
4195
4196 CaptureMouse();
4197 }
4198 }
4199 else if ( event.Moving() )
4200 {
4201 if ( item != m_dropTarget )
4202 {
4203 // unhighlight the previous drop target
4204 DrawDropEffect(m_dropTarget);
4205
4206 m_dropTarget = item;
4207
4208 // highlight the current drop target if any
4209 DrawDropEffect(m_dropTarget);
4210
4211 wxYieldIfNeeded();
4212 }
4213 }
4214 else if ( (event.LeftUp() || event.RightUp()) && m_isDragging )
4215 {
4216 // erase the highlighting
4217 DrawDropEffect(m_dropTarget);
4218
4219 if ( m_oldSelection )
4220 {
4221 m_oldSelection->SetHilight(TRUE);
4222 RefreshLine(m_oldSelection);
4223 m_oldSelection = (wxTreeListItem *)NULL;
4224 }
4225
4226 // generate the drag end event
4227 wxTreeEvent event(wxEVT_COMMAND_TREE_END_DRAG,/*ALB*/m_owner->GetId());
4228
4229 event.SetItem( (long) item );
4230 event.SetPoint( wxPoint(x, y) );
4231 event.SetEventObject(/*this*/m_owner);
4232
4233 (void)m_owner->GetEventHandler()->ProcessEvent(event);
4234
4235 m_isDragging = FALSE;
4236 m_dropTarget = (wxTreeListItem *)NULL;
4237
4238 ReleaseMouse();
4239
4240 SetCursor(m_oldCursor);
4241
4242 wxYieldIfNeeded();
4243 }
4244 else
4245 {
4246 // here we process only the messages which happen on tree items
4247
4248 m_dragCount = 0;
4249
4250 if ( item == NULL ) return; /* we hit the blank area */
4251
4252 if ( event.RightDown() )
4253 {
4254 SetFocus();
4255 wxTreeEvent nevent(wxEVT_COMMAND_TREE_ITEM_RIGHT_CLICK,
4256 m_owner->GetId());
4257 nevent.SetItem( (long) item );
4258 int nx, ny;
4259 CalcScrolledPosition(x, y, &nx, &ny);
4260 nevent.SetPoint( wxPoint(nx, ny));
4261 nevent.SetEventObject(/*this*/m_owner);
4262 m_owner->GetEventHandler()->ProcessEvent(nevent);
4263 }
4264 else if ( event.LeftUp() )
4265 {
4266 if ( m_lastOnSame )
4267 {
4268 if ( ( item == m_current ) &&
4269 ( flags & wxTREE_HITTEST_ONITEMLABEL ) &&
4270 HasFlag(wxTR_EDIT_LABELS ) )
4271 {
4272 if ( m_renameTimer->IsRunning() )
4273 m_renameTimer->Stop();
4274
4275 m_renameTimer->Start( 100, TRUE );
4276 }
4277
4278 m_lastOnSame = FALSE;
4279 }
4280 }
4281 else // !RightDown() && !LeftUp() ==> LeftDown() || LeftDClick()
4282 {
4283 if ( event.LeftDown() )
4284 {
4285 SetFocus();
4286 m_lastOnSame = item == m_current;
4287 }
4288
4289 if ((flags & wxTREE_HITTEST_ONITEMBUTTON) ||
4290 ((flags & wxTREE_HITTEST_ONITEMICON)) &&
4291 !HasButtons() && item->HasPlus())
4292 {
4293 // only toggle the item for a single click, double click on
4294 // the button doesn't do anything (it toggles the item twice)
4295 if ( event.LeftDown() )
4296 {
4297 Toggle( item );
4298 }
4299
4300 // don't select the item if the button was clicked
4301 return;
4302 }
4303
4304 // how should the selection work for this event?
4305 bool is_multiple, extended_select, unselect_others;
4306 EventFlagsToSelType(GetWindowStyleFlag(),
4307 event.ShiftDown(),
4308 event.ControlDown(),
4309 is_multiple, extended_select, unselect_others);
4310
4311 SelectItem (item, unselect_others, extended_select);
4312
4313 // For some reason, Windows isn't recognizing a left double-click,
4314 // so we need to simulate it here. Allow 200 milliseconds for now.
4315 if ( event.LeftDClick() )
4316 {
4317 // double clicking should not start editing the item label
4318 m_renameTimer->Stop();
4319 m_lastOnSame = FALSE;
4320
4321 // send activate event first
4322 wxTreeEvent nevent( wxEVT_COMMAND_TREE_ITEM_ACTIVATED,
4323 m_owner->GetId() );
4324 nevent.SetItem( (long) item );
4325 int nx, ny;
4326 CalcScrolledPosition(x, y, &nx, &ny);
4327 nevent.SetPoint( wxPoint(nx, ny) );
4328 nevent.SetEventObject( /*this*/m_owner );
4329 if ( !m_owner->GetEventHandler()->ProcessEvent( nevent ) )
4330 {
4331 // if the user code didn't process the activate event,
4332 // handle it ourselves by toggling the item when it is
4333 // double clicked
4334 if ( item->HasPlus() )
4335 {
4336 Toggle(item);
4337 }
4338 }
4339 }
4340 }
4341 }
4342 }
4343
4344 void wxTreeListMainWindow::OnIdle( wxIdleEvent &WXUNUSED(event) )
4345 {
4346 /* after all changes have been done to the tree control,
4347 * we actually redraw the tree when everything is over */
4348
4349 if (!m_dirty) return;
4350
4351 m_dirty = FALSE;
4352
4353 CalculatePositions();
4354 Refresh();
4355 AdjustMyScrollbars();
4356 }
4357
4358 void wxTreeListMainWindow::OnSize(wxSizeEvent& WXUNUSED(event))
4359 {
4360 // int w, h;
4361 // GetClientSize(&w, &h);
4362 // m_header_win->SetSize(0, 0, w, HEADER_HEIGHT);
4363 }
4364
4365 void wxTreeListMainWindow::OnScroll(wxScrollWinEvent& event)
4366 {
4367 // FIXME
4368 #if defined(__WXGTK__) && !defined(__WXUNIVERSAL__)
4369 wxScrolledWindow::OnScroll(event);
4370 #else
4371 HandleOnScroll( event );
4372 #endif
4373
4374 if(event.GetOrientation() == wxHORIZONTAL)
4375 {
4376 m_owner->GetHeaderWindow()->Refresh();
4377 m_owner->GetHeaderWindow()->Update();
4378 }
4379 }
4380
4381
4382 void wxTreeListMainWindow::CalculateSize( wxTreeListItem *item, wxDC &dc )
4383 {
4384 wxCoord text_w = 0;
4385 wxCoord text_h = 0;
4386
4387 if (item->IsBold())
4388 dc.SetFont(m_boldFont);
4389
4390 dc.GetTextExtent( item->GetText(/*ALB*/m_main_column), &text_w, &text_h );
4391 text_h+=2;
4392
4393 // restore normal font
4394 dc.SetFont( m_normalFont );
4395
4396 int total_h = (m_imgHeight > text_h) ? m_imgHeight : text_h;
4397
4398 item->SetHeight(total_h);
4399 if (total_h>m_lineHeight)
4400 m_lineHeight=total_h;
4401
4402 item->SetWidth(m_imgWidth + text_w+2);
4403 }
4404
4405 // -----------------------------------------------------------------------------
4406 // for developper : y is now the top of the level
4407 // not the middle of it !
4408 void wxTreeListMainWindow::CalculateLevel( wxTreeListItem *item, wxDC &dc,
4409 int level, int &y, int x_colstart )
4410 {
4411 // calculate position of vertical lines
4412 int x = x_colstart + MARGIN; // start of column
4413 if (HasFlag(wxTR_LINES_AT_ROOT)) x += LINEATROOT; // space for lines at root
4414 if (HasButtons()) x += m_btnWidth2; // space for buttons etc.
4415 if (!HasFlag(wxTR_HIDE_ROOT)) x += m_indent; // indent root as well
4416 x += m_indent * level; // indent according to level
4417
4418 // a hidden root is not evaluated, but its children are always
4419 if (HasFlag(wxTR_HIDE_ROOT) && (level == 0)) goto Recurse;
4420
4421 CalculateSize( item, dc );
4422
4423 // set its position
4424 item->SetX (x);
4425 item->SetY (y);
4426 y += GetLineHeight(item);
4427
4428 // we don't need to calculate collapsed branches
4429 if ( !item->IsExpanded() ) return;
4430
4431 Recurse:
4432 wxArrayTreeListItems& children = item->GetChildren();
4433 size_t n, count = children.Count();
4434 ++level;
4435 for (n = 0; n < count; ++n )
4436 CalculateLevel( children[n], dc, level, y, x_colstart ); // recurse
4437 }
4438
4439 void wxTreeListMainWindow::CalculatePositions()
4440 {
4441 if ( !m_anchor ) return;
4442
4443 wxClientDC dc(this);
4444 PrepareDC( dc );
4445
4446 dc.SetFont( m_normalFont );
4447
4448 dc.SetPen( m_dottedPen );
4449 //if(GetImageList() == NULL)
4450 // m_lineHeight = (int)(dc.GetCharHeight() + 4);
4451
4452 int y = 2;
4453 int x_colstart = 0;
4454 for(size_t i = 0; i < GetMainColumn(); ++i) {
4455 if (!m_owner->GetHeaderWindow()->GetColumnShown(i)) continue;
4456 x_colstart += m_owner->GetHeaderWindow()->GetColumnWidth(i);
4457 }
4458 CalculateLevel( m_anchor, dc, 0, y, x_colstart ); // start recursion
4459 }
4460
4461 void wxTreeListMainWindow::RefreshSubtree(wxTreeListItem *item)
4462 {
4463 if (m_dirty) return;
4464
4465 wxClientDC dc(this);
4466 PrepareDC(dc);
4467
4468 int cw = 0;
4469 int ch = 0;
4470 GetClientSize( &cw, &ch ); // GetVirtualSize???
4471
4472 wxRect rect;
4473 rect.x = dc.LogicalToDeviceX( 0 );
4474 rect.width = cw;
4475 rect.y = dc.LogicalToDeviceY( item->GetY() - 2 );
4476 rect.height = ch;
4477
4478 Refresh( TRUE, &rect );
4479
4480 AdjustMyScrollbars();
4481 }
4482
4483 void wxTreeListMainWindow::RefreshLine( wxTreeListItem *item )
4484 {
4485 if (m_dirty) return;
4486
4487 wxClientDC dc(this);
4488 PrepareDC( dc );
4489
4490 int cw = 0;
4491 int ch = 0;
4492 GetClientSize( &cw, &ch ); // GetVirtualSize ??
4493
4494 wxRect rect;
4495 rect.x = dc.LogicalToDeviceX( 0 );
4496 rect.y = dc.LogicalToDeviceY( item->GetY() );
4497 rect.width = cw;
4498 rect.height = GetLineHeight(item); //dc.GetCharHeight() + 6;
4499
4500 Refresh( TRUE, &rect );
4501 }
4502
4503 void wxTreeListMainWindow::RefreshSelected()
4504 {
4505 // TODO: this is awfully inefficient, we should keep the list of all
4506 // selected items internally, should be much faster
4507 if ( m_anchor )
4508 RefreshSelectedUnder(m_anchor);
4509 }
4510
4511 void wxTreeListMainWindow::RefreshSelectedUnder(wxTreeListItem *item)
4512 {
4513 if ( item->IsSelected() )
4514 RefreshLine(item);
4515
4516 const wxArrayTreeListItems& children = item->GetChildren();
4517 size_t count = children.GetCount();
4518 for ( size_t n = 0; n < count; n++ )
4519 {
4520 RefreshSelectedUnder(children[n]);
4521 }
4522 }
4523
4524 // ----------------------------------------------------------------------------
4525 // changing colours: we need to refresh the tree control
4526 // ----------------------------------------------------------------------------
4527
4528 bool wxTreeListMainWindow::SetBackgroundColour(const wxColour& colour)
4529 {
4530 if ( !wxWindow::SetBackgroundColour(colour) )
4531 return FALSE;
4532
4533 Refresh();
4534
4535 return TRUE;
4536 }
4537
4538 bool wxTreeListMainWindow::SetForegroundColour(const wxColour& colour)
4539 {
4540 if ( !wxWindow::SetForegroundColour(colour) )
4541 return FALSE;
4542
4543 Refresh();
4544
4545 return TRUE;
4546 }
4547
4548 //----------- ALB -------------
4549 void wxTreeListMainWindow::SetItemText(const wxTreeItemId& item, size_t column,
4550 const wxString& text)
4551 {
4552 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
4553
4554 wxClientDC dc(this);
4555 wxTreeListItem *pItem = (wxTreeListItem*) item.m_pItem;
4556 pItem->SetText(column, text);
4557 CalculateSize(pItem, dc);
4558 RefreshLine(pItem);
4559 }
4560
4561 wxString wxTreeListMainWindow::GetItemText(const wxTreeItemId& item,
4562 size_t column) const
4563 {
4564 wxCHECK_MSG( item.IsOk(), wxT(""), wxT("invalid tree item") );
4565
4566 return ((wxTreeListItem*) item.m_pItem)->GetText(column);
4567 }
4568
4569 void wxTreeListMainWindow::SetFocus()
4570 {
4571 wxWindow::SetFocus();
4572 }
4573
4574
4575 //-----------------------------------------------------------------------------
4576 // wxTreeListCtrl
4577 //-----------------------------------------------------------------------------
4578
4579 IMPLEMENT_DYNAMIC_CLASS(wxTreeListCtrl, wxControl);
4580
4581 BEGIN_EVENT_TABLE(wxTreeListCtrl, wxControl)
4582 EVT_SIZE(wxTreeListCtrl::OnSize)
4583 END_EVENT_TABLE();
4584
4585 bool wxTreeListCtrl::Create(wxWindow *parent, wxWindowID id,
4586 const wxPoint& pos,
4587 const wxSize& size,
4588 long style, const wxValidator &validator,
4589 const wxString& name)
4590 {
4591 long main_style = style & ~(wxRAISED_BORDER|wxSUNKEN_BORDER
4592 |wxSIMPLE_BORDER|wxNO_BORDER|wxDOUBLE_BORDER
4593 |wxSTATIC_BORDER);
4594 long ctrl_style = style & ~(wxVSCROLL|wxHSCROLL);
4595
4596 if (!wxControl::Create(parent, id, pos, size, ctrl_style, validator, name)) {
4597 return false;
4598 }
4599 m_main_win = new wxTreeListMainWindow(this, -1, wxPoint(0, 0), size,
4600 main_style, validator);
4601 m_header_win = new wxTreeListHeaderWindow(this, -1, m_main_win,
4602 wxPoint(0, 0), wxDefaultSize,
4603 wxTAB_TRAVERSAL);
4604 return TRUE;
4605 }
4606
4607 void wxTreeListCtrl::OnSize(wxSizeEvent& WXUNUSED(event))
4608 {
4609 int w, h;
4610 GetClientSize(&w, &h);
4611 if(m_header_win)
4612 m_header_win->SetSize(0, 0, w, HEADER_HEIGHT);
4613 if(m_main_win)
4614 m_main_win->SetSize(0, HEADER_HEIGHT + 1, w, h - HEADER_HEIGHT - 1);
4615 }
4616
4617
4618 size_t wxTreeListCtrl::GetCount() const { return m_main_win->GetCount(); }
4619
4620 unsigned int wxTreeListCtrl::GetIndent() const
4621 { return m_main_win->GetIndent(); }
4622
4623 void wxTreeListCtrl::SetIndent(unsigned int indent)
4624 { m_main_win->SetIndent(indent); }
4625
4626 unsigned int wxTreeListCtrl::GetLineSpacing() const
4627 { return m_main_win->GetLineSpacing(); }
4628
4629 void wxTreeListCtrl::SetLineSpacing(unsigned int spacing)
4630 { m_main_win->SetLineSpacing(spacing); }
4631
4632 wxImageList* wxTreeListCtrl::GetImageList() const
4633 { return m_main_win->GetImageList(); }
4634
4635 wxImageList* wxTreeListCtrl::GetStateImageList() const
4636 { return m_main_win->GetStateImageList(); }
4637
4638 wxImageList* wxTreeListCtrl::GetButtonsImageList() const
4639 { return m_main_win->GetButtonsImageList(); }
4640
4641 void wxTreeListCtrl::SetImageList(wxImageList* imageList)
4642 { m_main_win->SetImageList(imageList); }
4643
4644 void wxTreeListCtrl::SetStateImageList(wxImageList* imageList)
4645 { m_main_win->SetStateImageList(imageList); }
4646
4647 void wxTreeListCtrl::SetButtonsImageList(wxImageList* imageList)
4648 { m_main_win->SetButtonsImageList(imageList); }
4649
4650 void wxTreeListCtrl::AssignImageList(wxImageList* imageList)
4651 { m_main_win->AssignImageList(imageList); }
4652
4653 void wxTreeListCtrl::AssignStateImageList(wxImageList* imageList)
4654 { m_main_win->AssignStateImageList(imageList); }
4655
4656 void wxTreeListCtrl::AssignButtonsImageList(wxImageList* imageList)
4657 { m_main_win->AssignButtonsImageList(imageList); }
4658
4659 wxString wxTreeListCtrl::GetItemText(const wxTreeItemId& item, size_t column)
4660 const
4661 { return m_main_win->GetItemText(item, column); }
4662
4663 int wxTreeListCtrl::GetItemImage(const wxTreeItemId& item, size_t column,
4664 wxTreeItemIcon which) const
4665 { return m_main_win->GetItemImage(item, column, which); }
4666
4667 wxTreeItemData* wxTreeListCtrl::GetItemData(const wxTreeItemId& item) const
4668 { return m_main_win->GetItemData(item); }
4669
4670 bool wxTreeListCtrl::GetItemBold(const wxTreeItemId& item) const
4671 { return m_main_win->GetItemBold(item); }
4672
4673 wxColour wxTreeListCtrl::GetItemTextColour(const wxTreeItemId& item) const
4674 { return m_main_win->GetItemTextColour(item); }
4675
4676 wxColour wxTreeListCtrl::GetItemBackgroundColour(const wxTreeItemId& item)
4677 const
4678 { return m_main_win->GetItemBackgroundColour(item); }
4679
4680 wxFont wxTreeListCtrl::GetItemFont(const wxTreeItemId& item) const
4681 { return m_main_win->GetItemFont(item); }
4682
4683
4684 void wxTreeListCtrl::SetItemText(const wxTreeItemId& item, size_t column,
4685 const wxString& text)
4686 { m_main_win->SetItemText(item, column, text); }
4687
4688 void wxTreeListCtrl::SetItemImage(const wxTreeItemId& item,
4689 size_t column,
4690 int image,
4691 wxTreeItemIcon which)
4692 { m_main_win->SetItemImage(item, column, image, which); }
4693
4694 void wxTreeListCtrl::SetItemData(const wxTreeItemId& item,
4695 wxTreeItemData* data)
4696 { m_main_win->SetItemData(item, data); }
4697
4698 void wxTreeListCtrl::SetItemHasChildren(const wxTreeItemId& item, bool has)
4699 { m_main_win->SetItemHasChildren(item, has); }
4700
4701 void wxTreeListCtrl::SetItemBold(const wxTreeItemId& item, bool bold)
4702 { m_main_win->SetItemBold(item, bold); }
4703
4704 void wxTreeListCtrl::SetItemTextColour(const wxTreeItemId& item,
4705 const wxColour& col)
4706 { m_main_win->SetItemTextColour(item, col); }
4707
4708 void wxTreeListCtrl::SetItemBackgroundColour(const wxTreeItemId& item,
4709 const wxColour& col)
4710 { m_main_win->SetItemBackgroundColour(item, col); }
4711
4712 void wxTreeListCtrl::SetItemFont(const wxTreeItemId& item,
4713 const wxFont& font)
4714 { m_main_win->SetItemFont(item, font); }
4715
4716 bool wxTreeListCtrl::SetFont(const wxFont& font)
4717 {
4718 if(m_header_win) m_header_win->SetFont(font);
4719 if(m_main_win)
4720 return m_main_win->SetFont(font);
4721 else return FALSE;
4722 }
4723
4724 void wxTreeListCtrl::SetWindowStyle(const long style)
4725 {
4726 if(m_main_win)
4727 m_main_win->SetWindowStyle(style);
4728 // TODO: provide something like wxTL_NO_HEADERS to hide m_header_win
4729 }
4730
4731 long wxTreeListCtrl::GetWindowStyle() const
4732 {
4733 long style = m_windowStyle;
4734 if(m_main_win)
4735 style |= m_main_win->GetWindowStyle();
4736 return style;
4737 }
4738
4739 bool wxTreeListCtrl::IsVisible(const wxTreeItemId& item) const
4740 { return m_main_win->IsVisible(item); }
4741
4742 bool wxTreeListCtrl::ItemHasChildren(const wxTreeItemId& item) const
4743 { return m_main_win->ItemHasChildren(item); }
4744
4745 bool wxTreeListCtrl::IsExpanded(const wxTreeItemId& item) const
4746 { return m_main_win->IsExpanded(item); }
4747
4748 bool wxTreeListCtrl::IsSelected(const wxTreeItemId& item) const
4749 { return m_main_win->IsSelected(item); }
4750
4751 bool wxTreeListCtrl::IsBold(const wxTreeItemId& item) const
4752 { return m_main_win->IsBold(item); }
4753
4754 size_t wxTreeListCtrl::GetChildrenCount(const wxTreeItemId& item, bool rec)
4755 { return m_main_win->GetChildrenCount(item, rec); }
4756
4757 wxTreeItemId wxTreeListCtrl::GetRootItem() const
4758 { return m_main_win->GetRootItem(); }
4759
4760 wxTreeItemId wxTreeListCtrl::GetSelection() const
4761 { return m_main_win->GetSelection(); }
4762
4763 size_t wxTreeListCtrl::GetSelections(wxArrayTreeItemIds& arr) const
4764 { return m_main_win->GetSelections(arr); }
4765
4766 wxTreeItemId wxTreeListCtrl::GetItemParent(const wxTreeItemId& item) const
4767 { return m_main_win->GetItemParent(item); }
4768
4769 #if !wxCHECK_VERSION(2, 5, 0)
4770 wxTreeItemId wxTreeListCtrl::GetFirstChild(const wxTreeItemId& item,
4771 long& cookie) const
4772 #else
4773 wxTreeItemId wxTreeListCtrl::GetFirstChild(const wxTreeItemId& item,
4774 wxTreeItemIdValue& cookie) const
4775 #endif
4776 { return m_main_win->GetFirstChild(item, cookie); }
4777
4778 #if !wxCHECK_VERSION(2, 5, 0)
4779 wxTreeItemId wxTreeListCtrl::GetNextChild(const wxTreeItemId& item,
4780 long& cookie) const
4781 #else
4782 wxTreeItemId wxTreeListCtrl::GetNextChild(const wxTreeItemId& item,
4783 wxTreeItemIdValue& cookie) const
4784 #endif
4785 { return m_main_win->GetNextChild(item, cookie); }
4786
4787 #if !wxCHECK_VERSION(2, 5, 0)
4788 wxTreeItemId wxTreeListCtrl::GetPrevChild(const wxTreeItemId& item,
4789 long& cookie) const
4790 #else
4791 wxTreeItemId wxTreeListCtrl::GetPrevChild(const wxTreeItemId& item,
4792 wxTreeItemIdValue& cookie) const
4793 #endif
4794 { return m_main_win->GetPrevChild(item, cookie); }
4795
4796 wxTreeItemId wxTreeListCtrl::GetLastChild(const wxTreeItemId& item) const
4797 { return m_main_win->GetLastChild(item); }
4798
4799 wxTreeItemId wxTreeListCtrl::GetNextSibling(const wxTreeItemId& item) const
4800 { return m_main_win->GetNextSibling(item); }
4801
4802 wxTreeItemId wxTreeListCtrl::GetPrevSibling(const wxTreeItemId& item) const
4803 { return m_main_win->GetPrevSibling(item); }
4804
4805 wxTreeItemId wxTreeListCtrl::GetFirstVisibleItem() const
4806 { return m_main_win->GetFirstVisibleItem(); }
4807
4808 wxTreeItemId wxTreeListCtrl::GetNextVisible(const wxTreeItemId& item) const
4809 { return m_main_win->GetNextVisible(item); }
4810
4811 wxTreeItemId wxTreeListCtrl::GetPrevVisible(const wxTreeItemId& item) const
4812 { return m_main_win->GetPrevVisible(item); }
4813
4814 wxTreeItemId wxTreeListCtrl::GetNext(const wxTreeItemId& item) const
4815 { return m_main_win->GetNext(item); }
4816
4817 wxTreeItemId wxTreeListCtrl::AddRoot(const wxString& text, int image,
4818 int selectedImage, wxTreeItemData* data)
4819 { return m_main_win->AddRoot(text, image, selectedImage, data); }
4820
4821 wxTreeItemId wxTreeListCtrl::PrependItem(const wxTreeItemId& parent,
4822 const wxString& text, int image,
4823 int selectedImage,
4824 wxTreeItemData* data)
4825 { return m_main_win->PrependItem(parent, text, image, selectedImage, data); }
4826
4827 wxTreeItemId wxTreeListCtrl::InsertItem(const wxTreeItemId& parent,
4828 const wxTreeItemId& previous,
4829 const wxString& text, int image,
4830 int selectedImage,
4831 wxTreeItemData* data)
4832 {
4833 return m_main_win->InsertItem(parent, previous, text, image,
4834 selectedImage, data);
4835 }
4836
4837 wxTreeItemId wxTreeListCtrl::InsertItem(const wxTreeItemId& parent,
4838 size_t index,
4839 const wxString& text, int image,
4840 int selectedImage,
4841 wxTreeItemData* data)
4842 {
4843 return m_main_win->InsertItem(parent, index, text, image,
4844 selectedImage, data);
4845 }
4846
4847 wxTreeItemId wxTreeListCtrl::AppendItem(const wxTreeItemId& parent,
4848 const wxString& text, int image,
4849 int selectedImage,
4850 wxTreeItemData* data)
4851 { return m_main_win->AppendItem(parent, text, image, selectedImage, data); }
4852
4853 void wxTreeListCtrl::Delete(const wxTreeItemId& item)
4854 { m_main_win->Delete(item); }
4855
4856 void wxTreeListCtrl::DeleteChildren(const wxTreeItemId& item)
4857 { m_main_win->DeleteChildren(item); }
4858
4859 void wxTreeListCtrl::DeleteAllItems()
4860 { m_main_win->DeleteAllItems(); }
4861
4862 void wxTreeListCtrl::Expand(const wxTreeItemId& item)
4863 { m_main_win->Expand(item); }
4864
4865 void wxTreeListCtrl::ExpandAll(const wxTreeItemId& item)
4866 { m_main_win->ExpandAll(item); }
4867
4868 void wxTreeListCtrl::Collapse(const wxTreeItemId& item)
4869 { m_main_win->Collapse(item); }
4870
4871 void wxTreeListCtrl::CollapseAndReset(const wxTreeItemId& item)
4872 { m_main_win->CollapseAndReset(item); }
4873
4874 void wxTreeListCtrl::Toggle(const wxTreeItemId& item)
4875 { m_main_win->Toggle(item); }
4876
4877 void wxTreeListCtrl::Unselect()
4878 { m_main_win->Unselect(); }
4879
4880 void wxTreeListCtrl::UnselectAll()
4881 { m_main_win->UnselectAll(); }
4882
4883 void wxTreeListCtrl::SelectItem(const wxTreeItemId& item, bool unselect_others,
4884 bool extended_select)
4885 { m_main_win->SelectItem(item, unselect_others, extended_select); }
4886
4887 void wxTreeListCtrl::SelectAll(bool extended_select)
4888 { m_main_win->SelectAll(extended_select); }
4889
4890 void wxTreeListCtrl::EnsureVisible(const wxTreeItemId& item)
4891 { m_main_win->EnsureVisible(item); }
4892
4893 void wxTreeListCtrl::ScrollTo(const wxTreeItemId& item)
4894 { m_main_win->ScrollTo(item); }
4895
4896 wxTreeItemId wxTreeListCtrl::HitTest(const wxPoint& pos, int& flags,
4897 int& column)
4898 {
4899 return m_main_win->HitTest(m_main_win->ScreenToClient(ClientToScreen(pos)),
4900 flags, column);
4901 }
4902
4903 bool wxTreeListCtrl::GetBoundingRect(const wxTreeItemId& item, wxRect& rect,
4904 bool textOnly) const
4905 { return m_main_win->GetBoundingRect(item, rect, textOnly); }
4906
4907 void wxTreeListCtrl::Edit(const wxTreeItemId& item)
4908 { m_main_win->Edit(item); }
4909
4910 int wxTreeListCtrl::OnCompareItems(const wxTreeItemId& item1,
4911 const wxTreeItemId& item2)
4912 {
4913 // ALB: do the comparison here, and not delegate to m_main_win, in order
4914 // to let the user override it
4915 //return m_main_win->OnCompareItems(item1, item2);
4916 return wxStrcmp(GetItemText(item1), GetItemText(item2));
4917 }
4918
4919 void wxTreeListCtrl::SortChildren(const wxTreeItemId& item)
4920 { m_main_win->SortChildren(item); }
4921
4922 wxTreeItemId wxTreeListCtrl::FindItem (const wxTreeItemId& item, const wxString& str, int flags)
4923 { return m_main_win->FindItem (item, str, flags); }
4924
4925 bool wxTreeListCtrl::SetBackgroundColour(const wxColour& colour)
4926 { return m_main_win->SetBackgroundColour(colour); }
4927
4928 bool wxTreeListCtrl::SetForegroundColour(const wxColour& colour)
4929 { return m_main_win->SetForegroundColour(colour); }
4930
4931 size_t wxTreeListCtrl::GetColumnCount() const
4932 { return m_main_win->GetColumnCount(); }
4933
4934 void wxTreeListCtrl::SetColumnWidth(size_t column, size_t width)
4935 { m_header_win->SetColumnWidth(column, width); }
4936
4937 int wxTreeListCtrl::GetColumnWidth(size_t column) const
4938 { return m_header_win->GetColumnWidth(column); }
4939
4940 void wxTreeListCtrl::SetMainColumn(size_t column)
4941 { m_main_win->SetMainColumn(column); }
4942
4943 size_t wxTreeListCtrl::GetMainColumn() const
4944 { return m_main_win->GetMainColumn(); }
4945
4946 void wxTreeListCtrl::SetColumnText(size_t column, const wxString& text)
4947 {
4948 m_header_win->SetColumnText(column, text);
4949 m_header_win->Refresh();
4950 }
4951
4952 wxString wxTreeListCtrl::GetColumnText(size_t column) const
4953 { return m_header_win->GetColumnText(column); }
4954
4955 void wxTreeListCtrl::AddColumn(const wxTreeListColumnInfo& col)
4956 { m_header_win->AddColumn(col); }
4957
4958 void wxTreeListCtrl::InsertColumn(size_t before,
4959 const wxTreeListColumnInfo& col)
4960 { m_header_win->InsertColumn(before, col); }
4961
4962 void wxTreeListCtrl::RemoveColumn(size_t column)
4963 { m_header_win->RemoveColumn(column); }
4964
4965 void wxTreeListCtrl::SetColumn(size_t column, const wxTreeListColumnInfo& col)
4966 { m_header_win->SetColumn(column, col); }
4967
4968 const wxTreeListColumnInfo& wxTreeListCtrl::GetColumn(size_t column) const
4969 { return m_header_win->GetColumn(column); }
4970
4971 wxTreeListColumnInfo& wxTreeListCtrl::GetColumn(size_t column)
4972 { return m_header_win->GetColumn(column); }
4973
4974 void wxTreeListCtrl::SetColumnImage(size_t column, int image)
4975 {
4976 m_header_win->SetColumn(column, GetColumn(column).SetImage(image));
4977 }
4978
4979 int wxTreeListCtrl::GetColumnImage(size_t column) const
4980 {
4981 return m_header_win->GetColumn(column).GetImage();
4982 }
4983
4984 void wxTreeListCtrl::ShowColumn(size_t column, bool shown)
4985 {
4986 wxASSERT_MSG( column != GetMainColumn(),
4987 wxT("The main column may not be hidden") );
4988 m_header_win->SetColumn(column, GetColumn(column).SetShown(GetMainColumn()? true: shown));
4989 }
4990
4991 bool wxTreeListCtrl::IsColumnShown(size_t column) const
4992 {
4993 return m_header_win->GetColumn(column).GetShown();
4994 }
4995
4996 void wxTreeListCtrl::SetColumnAlignment(size_t column,
4997 wxTreeListColumnAlign align)
4998 {
4999 m_header_win->SetColumn(column, GetColumn(column).SetAlignment(align));
5000 }
5001
5002 wxTreeListColumnAlign wxTreeListCtrl::GetColumnAlignment(size_t column) const
5003 {
5004 return m_header_win->GetColumn(column).GetAlignment();
5005 }
5006
5007 void wxTreeListCtrl::Refresh(bool erase, const wxRect* rect)
5008 {
5009 m_main_win->Refresh(erase, rect);
5010 m_header_win->Refresh(erase, rect);
5011 }
5012
5013 void wxTreeListCtrl::SetFocus()
5014 { m_main_win->SetFocus(); }
5015