]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/treectrl.cpp
Add some version checks to help compiling on OSX.
[wxWidgets.git] / src / msw / treectrl.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/treectrl.cpp
3// Purpose: wxTreeCtrl
4// Author: Julian Smart
5// Modified by: Vadim Zeitlin to be less MSW-specific on 10.10.98
6// Created: 1997
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#if wxUSE_TREECTRL
28
29#include "wx/treectrl.h"
30
31#ifndef WX_PRECOMP
32 #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly"
33 #include "wx/msw/missing.h"
34 #include "wx/dynarray.h"
35 #include "wx/log.h"
36 #include "wx/app.h"
37 #include "wx/settings.h"
38#endif
39
40#include "wx/dynlib.h"
41#include "wx/msw/private.h"
42
43#include "wx/imaglist.h"
44#include "wx/msw/dragimag.h"
45#include "wx/msw/uxtheme.h"
46
47// macros to hide the cast ugliness
48// --------------------------------
49
50// get HTREEITEM from wxTreeItemId
51#define HITEM(item) ((HTREEITEM)(((item).m_pItem)))
52
53
54// older SDKs are missing these
55#ifndef TVN_ITEMCHANGINGA
56
57#define TVN_ITEMCHANGINGA (TVN_FIRST-16)
58#define TVN_ITEMCHANGINGW (TVN_FIRST-17)
59
60typedef struct tagNMTVITEMCHANGE
61{
62 NMHDR hdr;
63 UINT uChanged;
64 HTREEITEM hItem;
65 UINT uStateNew;
66 UINT uStateOld;
67 LPARAM lParam;
68} NMTVITEMCHANGE;
69
70#endif
71
72
73// this helper class is used on vista systems for preventing unwanted
74// item state changes in the vista tree control. It is only effective in
75// multi-select mode on vista systems.
76
77// The vista tree control includes some new code that originally broke the
78// multi-selection tree, causing seemingly spurious item selection state changes
79// during Shift or Ctrl-click item selection. (To witness the original broken
80// behaviour, simply make IsLocked() below always return false). This problem was
81// solved by using the following class to 'unlock' an item's selection state.
82
83class TreeItemUnlocker
84{
85public:
86 // unlock a single item
87 TreeItemUnlocker(HTREEITEM item) { ms_unlockedItem = item; }
88
89 // unlock all items, don't use unless absolutely necessary
90 TreeItemUnlocker() { ms_unlockedItem = (HTREEITEM)-1; }
91
92 // lock everything back
93 ~TreeItemUnlocker() { ms_unlockedItem = NULL; }
94
95
96 // check if the item state is currently locked
97 static bool IsLocked(HTREEITEM item)
98 { return ms_unlockedItem != (HTREEITEM)-1 && item != ms_unlockedItem; }
99
100private:
101 static HTREEITEM ms_unlockedItem;
102};
103
104HTREEITEM TreeItemUnlocker::ms_unlockedItem = NULL;
105
106// another helper class: set the variable to true during its lifetime and reset
107// it to false when it is destroyed
108//
109// it is currently always used with wxTreeCtrl::m_changingSelection
110class TempSetter
111{
112public:
113 TempSetter(bool& var) : m_var(var)
114 {
115 wxASSERT_MSG( !m_var, "variable shouldn't be already set" );
116 m_var = true;
117 }
118
119 ~TempSetter()
120 {
121 m_var = false;
122 }
123
124private:
125 bool& m_var;
126
127 wxDECLARE_NO_COPY_CLASS(TempSetter);
128};
129
130// ----------------------------------------------------------------------------
131// private functions
132// ----------------------------------------------------------------------------
133
134namespace
135{
136
137// Work around a problem with TreeView_GetItemRect() when using MinGW/Cygwin:
138// it results in warnings about breaking strict aliasing rules because HITEM is
139// passed via a RECT pointer, so use a union to avoid them and define our own
140// version of the standard macro using it.
141union TVGetItemRectParam
142{
143 RECT rect;
144 HTREEITEM hItem;
145};
146
147inline bool
148wxTreeView_GetItemRect(HWND hwnd,
149 HTREEITEM hItem,
150 TVGetItemRectParam& param,
151 BOOL fItemRect)
152{
153 param.hItem = hItem;
154 return ::SendMessage(hwnd, TVM_GETITEMRECT, fItemRect,
155 (LPARAM)&param) == TRUE;
156}
157
158} // anonymous namespace
159
160// wrappers for TreeView_GetItem/TreeView_SetItem
161static bool IsItemSelected(HWND hwndTV, HTREEITEM hItem)
162{
163 TV_ITEM tvi;
164 tvi.mask = TVIF_STATE | TVIF_HANDLE;
165 tvi.stateMask = TVIS_SELECTED;
166 tvi.hItem = hItem;
167
168 TreeItemUnlocker unlocker(hItem);
169
170 if ( !TreeView_GetItem(hwndTV, &tvi) )
171 {
172 wxLogLastError(wxT("TreeView_GetItem"));
173 }
174
175 return (tvi.state & TVIS_SELECTED) != 0;
176}
177
178static bool SelectItem(HWND hwndTV, HTREEITEM hItem, bool select = true)
179{
180 TV_ITEM tvi;
181 tvi.mask = TVIF_STATE | TVIF_HANDLE;
182 tvi.stateMask = TVIS_SELECTED;
183 tvi.state = select ? TVIS_SELECTED : 0;
184 tvi.hItem = hItem;
185
186 TreeItemUnlocker unlocker(hItem);
187
188 if ( TreeView_SetItem(hwndTV, &tvi) == -1 )
189 {
190 wxLogLastError(wxT("TreeView_SetItem"));
191 return false;
192 }
193
194 return true;
195}
196
197static inline void UnselectItem(HWND hwndTV, HTREEITEM htItem)
198{
199 SelectItem(hwndTV, htItem, false);
200}
201
202static inline void ToggleItemSelection(HWND hwndTV, HTREEITEM htItem)
203{
204 SelectItem(hwndTV, htItem, !IsItemSelected(hwndTV, htItem));
205}
206
207// helper function which selects all items in a range and, optionally,
208// deselects all the other ones
209//
210// returns true if the selection changed at all or false if nothing changed
211
212// flags for SelectRange()
213enum
214{
215 SR_SIMULATE = 1, // don't do anything, just return true or false
216 SR_UNSELECT_OTHERS = 2 // deselect the items not in range
217};
218
219static bool SelectRange(HWND hwndTV,
220 HTREEITEM htFirst,
221 HTREEITEM htLast,
222 int flags)
223{
224 // find the first (or last) item and select it
225 bool changed = false;
226 bool cont = true;
227 HTREEITEM htItem = (HTREEITEM)TreeView_GetRoot(hwndTV);
228
229 while ( htItem && cont )
230 {
231 if ( (htItem == htFirst) || (htItem == htLast) )
232 {
233 if ( !IsItemSelected(hwndTV, htItem) )
234 {
235 if ( !(flags & SR_SIMULATE) )
236 {
237 SelectItem(hwndTV, htItem);
238 }
239
240 changed = true;
241 }
242
243 cont = false;
244 }
245 else // not first or last
246 {
247 if ( flags & SR_UNSELECT_OTHERS )
248 {
249 if ( IsItemSelected(hwndTV, htItem) )
250 {
251 if ( !(flags & SR_SIMULATE) )
252 UnselectItem(hwndTV, htItem);
253
254 changed = true;
255 }
256 }
257 }
258
259 htItem = (HTREEITEM)TreeView_GetNextVisible(hwndTV, htItem);
260 }
261
262 // select the items in range
263 cont = htFirst != htLast;
264 while ( htItem && cont )
265 {
266 if ( !IsItemSelected(hwndTV, htItem) )
267 {
268 if ( !(flags & SR_SIMULATE) )
269 {
270 SelectItem(hwndTV, htItem);
271 }
272
273 changed = true;
274 }
275
276 cont = (htItem != htFirst) && (htItem != htLast);
277
278 htItem = (HTREEITEM)TreeView_GetNextVisible(hwndTV, htItem);
279 }
280
281 // optionally deselect the rest
282 if ( flags & SR_UNSELECT_OTHERS )
283 {
284 while ( htItem )
285 {
286 if ( IsItemSelected(hwndTV, htItem) )
287 {
288 if ( !(flags & SR_SIMULATE) )
289 {
290 UnselectItem(hwndTV, htItem);
291 }
292
293 changed = true;
294 }
295
296 htItem = (HTREEITEM)TreeView_GetNextVisible(hwndTV, htItem);
297 }
298 }
299
300 // seems to be necessary - otherwise the just selected items don't always
301 // appear as selected
302 if ( !(flags & SR_SIMULATE) )
303 {
304 UpdateWindow(hwndTV);
305 }
306
307 return changed;
308}
309
310// helper function which tricks the standard control into changing the focused
311// item without changing anything else (if someone knows why Microsoft doesn't
312// allow to do it by just setting TVIS_FOCUSED flag, please tell me!)
313//
314// returns true if the focus was changed, false if the given item was already
315// the focused one
316static bool SetFocus(HWND hwndTV, HTREEITEM htItem)
317{
318 // the current focus
319 HTREEITEM htFocus = (HTREEITEM)TreeView_GetSelection(hwndTV);
320
321 if ( htItem == htFocus )
322 return false;
323
324 if ( htItem )
325 {
326 // remember the selection state of the item
327 bool wasSelected = IsItemSelected(hwndTV, htItem);
328
329 if ( htFocus && IsItemSelected(hwndTV, htFocus) )
330 {
331 // prevent the tree from unselecting the old focus which it
332 // would do by default (TreeView_SelectItem unselects the
333 // focused item)
334 TreeView_SelectItem(hwndTV, 0);
335 SelectItem(hwndTV, htFocus);
336 }
337
338 TreeView_SelectItem(hwndTV, htItem);
339
340 if ( !wasSelected )
341 {
342 // need to clear the selection which TreeView_SelectItem() gave
343 // us
344 UnselectItem(hwndTV, htItem);
345 }
346 //else: was selected, still selected - ok
347 }
348 else // reset focus
349 {
350 bool wasFocusSelected = IsItemSelected(hwndTV, htFocus);
351
352 // just clear the focus
353 TreeView_SelectItem(hwndTV, 0);
354
355 if ( wasFocusSelected )
356 {
357 // restore the selection state
358 SelectItem(hwndTV, htFocus);
359 }
360 }
361
362 return true;
363}
364
365// ----------------------------------------------------------------------------
366// private classes
367// ----------------------------------------------------------------------------
368
369// a convenient wrapper around TV_ITEM struct which adds a ctor
370#ifdef __VISUALC__
371#pragma warning( disable : 4097 ) // inheriting from typedef
372#endif
373
374struct wxTreeViewItem : public TV_ITEM
375{
376 wxTreeViewItem(const wxTreeItemId& item, // the item handle
377 UINT mask_, // fields which are valid
378 UINT stateMask_ = 0) // for TVIF_STATE only
379 {
380 wxZeroMemory(*this);
381
382 // hItem member is always valid
383 mask = mask_ | TVIF_HANDLE;
384 stateMask = stateMask_;
385 hItem = HITEM(item);
386 }
387};
388
389// ----------------------------------------------------------------------------
390// This class is our userdata/lParam for the TV_ITEMs stored in the treeview.
391//
392// We need this for a couple of reasons:
393//
394// 1) This class is needed for support of different images: the Win32 common
395// control natively supports only 2 images (the normal one and another for the
396// selected state). We wish to provide support for 2 more of them for folder
397// items (i.e. those which have children): for expanded state and for expanded
398// selected state. For this we use this structure to store the additional items
399// images.
400//
401// 2) This class is also needed to hold the HITEM so that we can sort
402// it correctly in the MSW sort callback.
403//
404// In addition it makes other workarounds such as this easier and helps
405// simplify the code.
406// ----------------------------------------------------------------------------
407
408class wxTreeItemParam
409{
410public:
411 wxTreeItemParam()
412 {
413 m_data = NULL;
414
415 for ( size_t n = 0; n < WXSIZEOF(m_images); n++ )
416 {
417 m_images[n] = -1;
418 }
419 }
420
421 // dtor deletes the associated data as well
422 virtual ~wxTreeItemParam() { delete m_data; }
423
424 // accessors
425 // get the real data associated with the item
426 wxTreeItemData *GetData() const { return m_data; }
427 // change it
428 void SetData(wxTreeItemData *data) { m_data = data; }
429
430 // do we have such image?
431 bool HasImage(wxTreeItemIcon which) const { return m_images[which] != -1; }
432 // get image, falling back to the other images if this one is not
433 // specified
434 int GetImage(wxTreeItemIcon which) const
435 {
436 int image = m_images[which];
437 if ( image == -1 )
438 {
439 switch ( which )
440 {
441 case wxTreeItemIcon_SelectedExpanded:
442 image = GetImage(wxTreeItemIcon_Expanded);
443 if ( image != -1 )
444 break;
445 //else: fall through
446
447 case wxTreeItemIcon_Selected:
448 case wxTreeItemIcon_Expanded:
449 image = GetImage(wxTreeItemIcon_Normal);
450 break;
451
452 case wxTreeItemIcon_Normal:
453 // no fallback
454 break;
455
456 default:
457 wxFAIL_MSG( wxT("unsupported wxTreeItemIcon value") );
458 }
459 }
460
461 return image;
462 }
463 // change the given image
464 void SetImage(int image, wxTreeItemIcon which) { m_images[which] = image; }
465
466 // get item
467 const wxTreeItemId& GetItem() const { return m_item; }
468 // set item
469 void SetItem(const wxTreeItemId& item) { m_item = item; }
470
471protected:
472 // all the images associated with the item
473 int m_images[wxTreeItemIcon_Max];
474
475 // item for sort callbacks
476 wxTreeItemId m_item;
477
478 // the real client data
479 wxTreeItemData *m_data;
480
481 wxDECLARE_NO_COPY_CLASS(wxTreeItemParam);
482};
483
484// wxVirutalNode is used in place of a single root when 'hidden' root is
485// specified.
486class wxVirtualNode : public wxTreeViewItem
487{
488public:
489 wxVirtualNode(wxTreeItemParam *param)
490 : wxTreeViewItem(TVI_ROOT, 0)
491 {
492 m_param = param;
493 }
494
495 ~wxVirtualNode()
496 {
497 delete m_param;
498 }
499
500 wxTreeItemParam *GetParam() const { return m_param; }
501 void SetParam(wxTreeItemParam *param) { delete m_param; m_param = param; }
502
503private:
504 wxTreeItemParam *m_param;
505
506 wxDECLARE_NO_COPY_CLASS(wxVirtualNode);
507};
508
509#ifdef __VISUALC__
510#pragma warning( default : 4097 )
511#endif
512
513// a macro to get the virtual root, returns NULL if none
514#define GET_VIRTUAL_ROOT() ((wxVirtualNode *)m_pVirtualRoot)
515
516// returns true if the item is the virtual root
517#define IS_VIRTUAL_ROOT(item) (HITEM(item) == TVI_ROOT)
518
519// a class which encapsulates the tree traversal logic: it vists all (unless
520// OnVisit() returns false) items under the given one
521class wxTreeTraversal
522{
523public:
524 wxTreeTraversal(const wxTreeCtrl *tree)
525 {
526 m_tree = tree;
527 }
528
529 // give it a virtual dtor: not really needed as the class is never used
530 // polymorphically and not even allocated on heap at all, but this is safer
531 // (in case it ever is) and silences the compiler warnings for now
532 virtual ~wxTreeTraversal() { }
533
534 // do traverse the tree: visit all items (recursively by default) under the
535 // given one; return true if all items were traversed or false if the
536 // traversal was aborted because OnVisit returned false
537 bool DoTraverse(const wxTreeItemId& root, bool recursively = true);
538
539 // override this function to do whatever is needed for each item, return
540 // false to stop traversing
541 virtual bool OnVisit(const wxTreeItemId& item) = 0;
542
543protected:
544 const wxTreeCtrl *GetTree() const { return m_tree; }
545
546private:
547 bool Traverse(const wxTreeItemId& root, bool recursively);
548
549 const wxTreeCtrl *m_tree;
550
551 wxDECLARE_NO_COPY_CLASS(wxTreeTraversal);
552};
553
554// internal class for getting the selected items
555class TraverseSelections : public wxTreeTraversal
556{
557public:
558 TraverseSelections(const wxTreeCtrl *tree,
559 wxArrayTreeItemIds& selections)
560 : wxTreeTraversal(tree), m_selections(selections)
561 {
562 m_selections.Empty();
563
564 if (tree->GetCount() > 0)
565 DoTraverse(tree->GetRootItem());
566 }
567
568 virtual bool OnVisit(const wxTreeItemId& item)
569 {
570 const wxTreeCtrl * const tree = GetTree();
571
572 // can't visit a virtual node.
573 if ( (tree->GetRootItem() == item) && tree->HasFlag(wxTR_HIDE_ROOT) )
574 {
575 return true;
576 }
577
578 if ( ::IsItemSelected(GetHwndOf(tree), HITEM(item)) )
579 {
580 m_selections.Add(item);
581 }
582
583 return true;
584 }
585
586 size_t GetCount() const { return m_selections.GetCount(); }
587
588private:
589 wxArrayTreeItemIds& m_selections;
590
591 wxDECLARE_NO_COPY_CLASS(TraverseSelections);
592};
593
594// internal class for counting tree items
595class TraverseCounter : public wxTreeTraversal
596{
597public:
598 TraverseCounter(const wxTreeCtrl *tree,
599 const wxTreeItemId& root,
600 bool recursively)
601 : wxTreeTraversal(tree)
602 {
603 m_count = 0;
604
605 DoTraverse(root, recursively);
606 }
607
608 virtual bool OnVisit(const wxTreeItemId& WXUNUSED(item))
609 {
610 m_count++;
611
612 return true;
613 }
614
615 size_t GetCount() const { return m_count; }
616
617private:
618 size_t m_count;
619
620 wxDECLARE_NO_COPY_CLASS(TraverseCounter);
621};
622
623// ----------------------------------------------------------------------------
624// wxWin macros
625// ----------------------------------------------------------------------------
626
627// ----------------------------------------------------------------------------
628// constants
629// ----------------------------------------------------------------------------
630
631// indices in gs_expandEvents table below
632enum
633{
634 IDX_COLLAPSE,
635 IDX_EXPAND,
636 IDX_WHAT_MAX
637};
638
639enum
640{
641 IDX_DONE,
642 IDX_DOING,
643 IDX_HOW_MAX
644};
645
646// handy table for sending events - it has to be initialized during run-time
647// now so can't be const any more
648static /* const */ wxEventType gs_expandEvents[IDX_WHAT_MAX][IDX_HOW_MAX];
649
650/*
651 but logically it's a const table with the following entries:
652=
653{
654 { wxEVT_COMMAND_TREE_ITEM_COLLAPSED, wxEVT_COMMAND_TREE_ITEM_COLLAPSING },
655 { wxEVT_COMMAND_TREE_ITEM_EXPANDED, wxEVT_COMMAND_TREE_ITEM_EXPANDING }
656};
657*/
658
659// ============================================================================
660// implementation
661// ============================================================================
662
663// ----------------------------------------------------------------------------
664// tree traversal
665// ----------------------------------------------------------------------------
666
667bool wxTreeTraversal::DoTraverse(const wxTreeItemId& root, bool recursively)
668{
669 if ( !OnVisit(root) )
670 return false;
671
672 return Traverse(root, recursively);
673}
674
675bool wxTreeTraversal::Traverse(const wxTreeItemId& root, bool recursively)
676{
677 wxTreeItemIdValue cookie;
678 wxTreeItemId child = m_tree->GetFirstChild(root, cookie);
679 while ( child.IsOk() )
680 {
681 // depth first traversal
682 if ( recursively && !Traverse(child, true) )
683 return false;
684
685 if ( !OnVisit(child) )
686 return false;
687
688 child = m_tree->GetNextChild(root, cookie);
689 }
690
691 return true;
692}
693
694// ----------------------------------------------------------------------------
695// construction and destruction
696// ----------------------------------------------------------------------------
697
698void wxTreeCtrl::Init()
699{
700 m_textCtrl = NULL;
701 m_hasAnyAttr = false;
702#if wxUSE_DRAGIMAGE
703 m_dragImage = NULL;
704#endif
705 m_pVirtualRoot = NULL;
706 m_dragStarted = false;
707 m_focusLost = true;
708 m_changingSelection = false;
709 m_triggerStateImageClick = false;
710 m_mouseUpDeselect = false;
711
712 // initialize the global array of events now as it can't be done statically
713 // with the wxEVT_XXX values being allocated during run-time only
714 gs_expandEvents[IDX_COLLAPSE][IDX_DONE] = wxEVT_COMMAND_TREE_ITEM_COLLAPSED;
715 gs_expandEvents[IDX_COLLAPSE][IDX_DOING] = wxEVT_COMMAND_TREE_ITEM_COLLAPSING;
716 gs_expandEvents[IDX_EXPAND][IDX_DONE] = wxEVT_COMMAND_TREE_ITEM_EXPANDED;
717 gs_expandEvents[IDX_EXPAND][IDX_DOING] = wxEVT_COMMAND_TREE_ITEM_EXPANDING;
718}
719
720bool wxTreeCtrl::Create(wxWindow *parent,
721 wxWindowID id,
722 const wxPoint& pos,
723 const wxSize& size,
724 long style,
725 const wxValidator& validator,
726 const wxString& name)
727{
728 Init();
729
730 if ( (style & wxBORDER_MASK) == wxBORDER_DEFAULT )
731 style |= wxBORDER_SUNKEN;
732
733 if ( !CreateControl(parent, id, pos, size, style, validator, name) )
734 return false;
735
736 WXDWORD exStyle = 0;
737 DWORD wstyle = MSWGetStyle(m_windowStyle, & exStyle);
738 wstyle |= WS_TABSTOP | TVS_SHOWSELALWAYS;
739
740 if ( !(m_windowStyle & wxTR_NO_LINES) )
741 wstyle |= TVS_HASLINES;
742 if ( m_windowStyle & wxTR_HAS_BUTTONS )
743 wstyle |= TVS_HASBUTTONS;
744
745 if ( m_windowStyle & wxTR_EDIT_LABELS )
746 wstyle |= TVS_EDITLABELS;
747
748 if ( m_windowStyle & wxTR_LINES_AT_ROOT )
749 wstyle |= TVS_LINESATROOT;
750
751 if ( m_windowStyle & wxTR_FULL_ROW_HIGHLIGHT )
752 {
753 if ( wxApp::GetComCtl32Version() >= 471 )
754 wstyle |= TVS_FULLROWSELECT;
755 }
756
757#if !defined(__WXWINCE__) && defined(TVS_INFOTIP)
758 // Need so that TVN_GETINFOTIP messages will be sent
759 wstyle |= TVS_INFOTIP;
760#endif
761
762 // Create the tree control.
763 if ( !MSWCreateControl(WC_TREEVIEW, wstyle, pos, size) )
764 return false;
765
766 SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW));
767 SetForegroundColour(wxWindow::GetParent()->GetForegroundColour());
768
769 wxSetCCUnicodeFormat(GetHwnd());
770
771 if ( m_windowStyle & wxTR_TWIST_BUTTONS )
772 {
773 // Under Vista and later Explorer uses rotating ("twist") buttons
774 // instead of the default "+/-" ones so apply its theme to the tree
775 // control to implement this style.
776 if ( wxGetWinVersion() >= wxWinVersion_Vista )
777 {
778 if ( wxUxThemeEngine *theme = wxUxThemeEngine::GetIfActive() )
779 {
780 theme->SetWindowTheme(GetHwnd(), L"EXPLORER", NULL);
781 }
782 }
783 }
784
785 return true;
786}
787
788wxTreeCtrl::~wxTreeCtrl()
789{
790 // delete any attributes
791 if ( m_hasAnyAttr )
792 {
793 WX_CLEAR_HASH_MAP(wxMapTreeAttr, m_attrs);
794
795 // prevent TVN_DELETEITEM handler from deleting the attributes again!
796 m_hasAnyAttr = false;
797 }
798
799 DeleteTextCtrl();
800
801 // delete user data to prevent memory leaks
802 // also deletes hidden root node storage.
803 DeleteAllItems();
804}
805
806// ----------------------------------------------------------------------------
807// accessors
808// ----------------------------------------------------------------------------
809
810/* static */ wxVisualAttributes
811wxTreeCtrl::GetClassDefaultAttributes(wxWindowVariant variant)
812{
813 wxVisualAttributes attrs = GetCompositeControlsDefaultAttributes(variant);
814
815 // common controls have their own default font
816 attrs.font = wxGetCCDefaultFont();
817
818 return attrs;
819}
820
821
822// simple wrappers which add error checking in debug mode
823
824bool wxTreeCtrl::DoGetItem(wxTreeViewItem *tvItem) const
825{
826 wxCHECK_MSG( tvItem->hItem != TVI_ROOT, false,
827 wxT("can't retrieve virtual root item") );
828
829 if ( !TreeView_GetItem(GetHwnd(), tvItem) )
830 {
831 wxLogLastError(wxT("TreeView_GetItem"));
832
833 return false;
834 }
835
836 return true;
837}
838
839void wxTreeCtrl::DoSetItem(wxTreeViewItem *tvItem)
840{
841 TreeItemUnlocker unlocker(tvItem->hItem);
842
843 if ( TreeView_SetItem(GetHwnd(), tvItem) == -1 )
844 {
845 wxLogLastError(wxT("TreeView_SetItem"));
846 }
847}
848
849unsigned int wxTreeCtrl::GetCount() const
850{
851 return (unsigned int)TreeView_GetCount(GetHwnd());
852}
853
854unsigned int wxTreeCtrl::GetIndent() const
855{
856 return TreeView_GetIndent(GetHwnd());
857}
858
859void wxTreeCtrl::SetIndent(unsigned int indent)
860{
861 TreeView_SetIndent(GetHwnd(), indent);
862}
863
864void wxTreeCtrl::SetAnyImageList(wxImageList *imageList, int which)
865{
866 // no error return
867 (void) TreeView_SetImageList(GetHwnd(),
868 imageList ? imageList->GetHIMAGELIST() : 0,
869 which);
870}
871
872void wxTreeCtrl::SetImageList(wxImageList *imageList)
873{
874 if (m_ownsImageListNormal)
875 delete m_imageListNormal;
876
877 SetAnyImageList(m_imageListNormal = imageList, TVSIL_NORMAL);
878 m_ownsImageListNormal = false;
879}
880
881void wxTreeCtrl::SetStateImageList(wxImageList *imageList)
882{
883 if (m_ownsImageListState) delete m_imageListState;
884 SetAnyImageList(m_imageListState = imageList, TVSIL_STATE);
885 m_ownsImageListState = false;
886}
887
888size_t wxTreeCtrl::GetChildrenCount(const wxTreeItemId& item,
889 bool recursively) const
890{
891 wxCHECK_MSG( item.IsOk(), 0u, wxT("invalid tree item") );
892
893 TraverseCounter counter(this, item, recursively);
894 return counter.GetCount() - 1;
895}
896
897// ----------------------------------------------------------------------------
898// control colours
899// ----------------------------------------------------------------------------
900
901bool wxTreeCtrl::SetBackgroundColour(const wxColour &colour)
902{
903 if ( !wxWindowBase::SetBackgroundColour(colour) )
904 return false;
905
906 ::SendMessage(GetHwnd(), TVM_SETBKCOLOR, 0, colour.GetPixel());
907
908 return true;
909}
910
911bool wxTreeCtrl::SetForegroundColour(const wxColour &colour)
912{
913 if ( !wxWindowBase::SetForegroundColour(colour) )
914 return false;
915
916 ::SendMessage(GetHwnd(), TVM_SETTEXTCOLOR, 0, colour.GetPixel());
917
918 return true;
919}
920
921// ----------------------------------------------------------------------------
922// Item access
923// ----------------------------------------------------------------------------
924
925bool wxTreeCtrl::IsHiddenRoot(const wxTreeItemId& item) const
926{
927 return HITEM(item) == TVI_ROOT && HasFlag(wxTR_HIDE_ROOT);
928}
929
930wxString wxTreeCtrl::GetItemText(const wxTreeItemId& item) const
931{
932 wxCHECK_MSG( item.IsOk(), wxEmptyString, wxT("invalid tree item") );
933
934 wxChar buf[512]; // the size is arbitrary...
935
936 wxTreeViewItem tvItem(item, TVIF_TEXT);
937 tvItem.pszText = buf;
938 tvItem.cchTextMax = WXSIZEOF(buf);
939 if ( !DoGetItem(&tvItem) )
940 {
941 // don't return some garbage which was on stack, but an empty string
942 buf[0] = wxT('\0');
943 }
944
945 return wxString(buf);
946}
947
948void wxTreeCtrl::SetItemText(const wxTreeItemId& item, const wxString& text)
949{
950 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
951
952 if ( IS_VIRTUAL_ROOT(item) )
953 return;
954
955 wxTreeViewItem tvItem(item, TVIF_TEXT);
956 tvItem.pszText = (wxChar *)text.wx_str(); // conversion is ok
957 DoSetItem(&tvItem);
958
959 // when setting the text of the item being edited, the text control should
960 // be updated to reflect the new text as well, otherwise calling
961 // SetItemText() in the OnBeginLabelEdit() handler doesn't have any effect
962 //
963 // don't use GetEditControl() here because m_textCtrl is not set yet
964 HWND hwndEdit = TreeView_GetEditControl(GetHwnd());
965 if ( hwndEdit )
966 {
967 if ( item == m_idEdited )
968 {
969 ::SetWindowText(hwndEdit, text.wx_str());
970 }
971 }
972}
973
974int wxTreeCtrl::GetItemImage(const wxTreeItemId& item,
975 wxTreeItemIcon which) const
976{
977 wxCHECK_MSG( item.IsOk(), -1, wxT("invalid tree item") );
978
979 if ( IsHiddenRoot(item) )
980 {
981 // no images for hidden root item
982 return -1;
983 }
984
985 wxTreeItemParam *param = GetItemParam(item);
986
987 return param && param->HasImage(which) ? param->GetImage(which) : -1;
988}
989
990void wxTreeCtrl::SetItemImage(const wxTreeItemId& item, int image,
991 wxTreeItemIcon which)
992{
993 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
994 wxCHECK_RET( which >= 0 &&
995 which < wxTreeItemIcon_Max,
996 wxT("invalid image index"));
997
998
999 if ( IsHiddenRoot(item) )
1000 {
1001 // no images for hidden root item
1002 return;
1003 }
1004
1005 wxTreeItemParam *data = GetItemParam(item);
1006 if ( !data )
1007 return;
1008
1009 data->SetImage(image, which);
1010
1011 RefreshItem(item);
1012}
1013
1014wxTreeItemParam *wxTreeCtrl::GetItemParam(const wxTreeItemId& item) const
1015{
1016 wxCHECK_MSG( item.IsOk(), NULL, wxT("invalid tree item") );
1017
1018 wxTreeViewItem tvItem(item, TVIF_PARAM);
1019
1020 // hidden root may still have data.
1021 if ( IS_VIRTUAL_ROOT(item) )
1022 {
1023 return GET_VIRTUAL_ROOT()->GetParam();
1024 }
1025
1026 // visible node.
1027 if ( !DoGetItem(&tvItem) )
1028 {
1029 return NULL;
1030 }
1031
1032 return (wxTreeItemParam *)tvItem.lParam;
1033}
1034
1035bool wxTreeCtrl::HandleTreeEvent(wxTreeEvent& event) const
1036{
1037 if ( event.m_item.IsOk() )
1038 {
1039 event.SetClientObject(GetItemData(event.m_item));
1040 }
1041
1042 return HandleWindowEvent(event);
1043}
1044
1045wxTreeItemData *wxTreeCtrl::GetItemData(const wxTreeItemId& item) const
1046{
1047 wxTreeItemParam *data = GetItemParam(item);
1048
1049 return data ? data->GetData() : NULL;
1050}
1051
1052void wxTreeCtrl::SetItemData(const wxTreeItemId& item, wxTreeItemData *data)
1053{
1054 // first, associate this piece of data with this item
1055 if ( data )
1056 {
1057 data->SetId(item);
1058 }
1059
1060 wxTreeItemParam *param = GetItemParam(item);
1061
1062 wxCHECK_RET( param, wxT("failed to change tree items data") );
1063
1064 param->SetData(data);
1065}
1066
1067void wxTreeCtrl::SetItemHasChildren(const wxTreeItemId& item, bool has)
1068{
1069 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
1070
1071 if ( IS_VIRTUAL_ROOT(item) )
1072 return;
1073
1074 wxTreeViewItem tvItem(item, TVIF_CHILDREN);
1075 tvItem.cChildren = (int)has;
1076 DoSetItem(&tvItem);
1077}
1078
1079void wxTreeCtrl::SetItemBold(const wxTreeItemId& item, bool bold)
1080{
1081 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
1082
1083 if ( IS_VIRTUAL_ROOT(item) )
1084 return;
1085
1086 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_BOLD);
1087 tvItem.state = bold ? TVIS_BOLD : 0;
1088 DoSetItem(&tvItem);
1089}
1090
1091void wxTreeCtrl::SetItemDropHighlight(const wxTreeItemId& item, bool highlight)
1092{
1093 if ( IS_VIRTUAL_ROOT(item) )
1094 return;
1095
1096 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_DROPHILITED);
1097 tvItem.state = highlight ? TVIS_DROPHILITED : 0;
1098 DoSetItem(&tvItem);
1099}
1100
1101void wxTreeCtrl::RefreshItem(const wxTreeItemId& item)
1102{
1103 if ( IS_VIRTUAL_ROOT(item) )
1104 return;
1105
1106 wxRect rect;
1107 if ( GetBoundingRect(item, rect) )
1108 {
1109 RefreshRect(rect);
1110 }
1111}
1112
1113wxColour wxTreeCtrl::GetItemTextColour(const wxTreeItemId& item) const
1114{
1115 wxCHECK_MSG( item.IsOk(), wxNullColour, wxT("invalid tree item") );
1116
1117 wxMapTreeAttr::const_iterator it = m_attrs.find(item.m_pItem);
1118 return it == m_attrs.end() ? wxNullColour : it->second->GetTextColour();
1119}
1120
1121wxColour wxTreeCtrl::GetItemBackgroundColour(const wxTreeItemId& item) const
1122{
1123 wxCHECK_MSG( item.IsOk(), wxNullColour, wxT("invalid tree item") );
1124
1125 wxMapTreeAttr::const_iterator it = m_attrs.find(item.m_pItem);
1126 return it == m_attrs.end() ? wxNullColour : it->second->GetBackgroundColour();
1127}
1128
1129wxFont wxTreeCtrl::GetItemFont(const wxTreeItemId& item) const
1130{
1131 wxCHECK_MSG( item.IsOk(), wxNullFont, wxT("invalid tree item") );
1132
1133 wxMapTreeAttr::const_iterator it = m_attrs.find(item.m_pItem);
1134 return it == m_attrs.end() ? wxNullFont : it->second->GetFont();
1135}
1136
1137void wxTreeCtrl::SetItemTextColour(const wxTreeItemId& item,
1138 const wxColour& col)
1139{
1140 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
1141
1142 wxTreeItemAttr *attr;
1143 wxMapTreeAttr::iterator it = m_attrs.find(item.m_pItem);
1144 if ( it == m_attrs.end() )
1145 {
1146 m_hasAnyAttr = true;
1147
1148 m_attrs[item.m_pItem] =
1149 attr = new wxTreeItemAttr;
1150 }
1151 else
1152 {
1153 attr = it->second;
1154 }
1155
1156 attr->SetTextColour(col);
1157
1158 RefreshItem(item);
1159}
1160
1161void wxTreeCtrl::SetItemBackgroundColour(const wxTreeItemId& item,
1162 const wxColour& col)
1163{
1164 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
1165
1166 wxTreeItemAttr *attr;
1167 wxMapTreeAttr::iterator it = m_attrs.find(item.m_pItem);
1168 if ( it == m_attrs.end() )
1169 {
1170 m_hasAnyAttr = true;
1171
1172 m_attrs[item.m_pItem] =
1173 attr = new wxTreeItemAttr;
1174 }
1175 else // already in the hash
1176 {
1177 attr = it->second;
1178 }
1179
1180 attr->SetBackgroundColour(col);
1181
1182 RefreshItem(item);
1183}
1184
1185void wxTreeCtrl::SetItemFont(const wxTreeItemId& item, const wxFont& font)
1186{
1187 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
1188
1189 wxTreeItemAttr *attr;
1190 wxMapTreeAttr::iterator it = m_attrs.find(item.m_pItem);
1191 if ( it == m_attrs.end() )
1192 {
1193 m_hasAnyAttr = true;
1194
1195 m_attrs[item.m_pItem] =
1196 attr = new wxTreeItemAttr;
1197 }
1198 else // already in the hash
1199 {
1200 attr = it->second;
1201 }
1202
1203 attr->SetFont(font);
1204
1205 // Reset the item's text to ensure that the bounding rect will be adjusted
1206 // for the new font.
1207 SetItemText(item, GetItemText(item));
1208
1209 RefreshItem(item);
1210}
1211
1212// ----------------------------------------------------------------------------
1213// Item status
1214// ----------------------------------------------------------------------------
1215
1216bool wxTreeCtrl::IsVisible(const wxTreeItemId& item) const
1217{
1218 wxCHECK_MSG( item.IsOk(), false, wxT("invalid tree item") );
1219
1220 if ( item == wxTreeItemId(TVI_ROOT) )
1221 {
1222 // virtual (hidden) root is never visible
1223 return false;
1224 }
1225
1226 // Bug in Gnu-Win32 headers, so don't use the macro TreeView_GetItemRect
1227 TVGetItemRectParam param;
1228
1229 // true means to get rect for just the text, not the whole line
1230 if ( !wxTreeView_GetItemRect(GetHwnd(), HITEM(item), param, TRUE) )
1231 {
1232 // if TVM_GETITEMRECT returned false, then the item is definitely not
1233 // visible (because its parent is not expanded)
1234 return false;
1235 }
1236
1237 // however if it returned true, the item might still be outside the
1238 // currently visible part of the tree, test for it (notice that partly
1239 // visible means visible here)
1240 return param.rect.bottom > 0 && param.rect.top < GetClientSize().y;
1241}
1242
1243bool wxTreeCtrl::ItemHasChildren(const wxTreeItemId& item) const
1244{
1245 wxCHECK_MSG( item.IsOk(), false, wxT("invalid tree item") );
1246
1247 if ( IS_VIRTUAL_ROOT(item) )
1248 {
1249 wxTreeItemIdValue cookie;
1250 return GetFirstChild(item, cookie).IsOk();
1251 }
1252
1253 wxTreeViewItem tvItem(item, TVIF_CHILDREN);
1254 DoGetItem(&tvItem);
1255
1256 return tvItem.cChildren != 0;
1257}
1258
1259bool wxTreeCtrl::IsExpanded(const wxTreeItemId& item) const
1260{
1261 wxCHECK_MSG( item.IsOk(), false, wxT("invalid tree item") );
1262
1263 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_EXPANDED);
1264 DoGetItem(&tvItem);
1265
1266 return (tvItem.state & TVIS_EXPANDED) != 0;
1267}
1268
1269bool wxTreeCtrl::IsSelected(const wxTreeItemId& item) const
1270{
1271 wxCHECK_MSG( item.IsOk(), false, wxT("invalid tree item") );
1272
1273 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_SELECTED);
1274 DoGetItem(&tvItem);
1275
1276 return (tvItem.state & TVIS_SELECTED) != 0;
1277}
1278
1279bool wxTreeCtrl::IsBold(const wxTreeItemId& item) const
1280{
1281 wxCHECK_MSG( item.IsOk(), false, wxT("invalid tree item") );
1282
1283 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_BOLD);
1284 DoGetItem(&tvItem);
1285
1286 return (tvItem.state & TVIS_BOLD) != 0;
1287}
1288
1289// ----------------------------------------------------------------------------
1290// navigation
1291// ----------------------------------------------------------------------------
1292
1293wxTreeItemId wxTreeCtrl::GetRootItem() const
1294{
1295 // Root may be real (visible) or virtual (hidden).
1296 if ( GET_VIRTUAL_ROOT() )
1297 return TVI_ROOT;
1298
1299 return wxTreeItemId(TreeView_GetRoot(GetHwnd()));
1300}
1301
1302wxTreeItemId wxTreeCtrl::GetSelection() const
1303{
1304 wxCHECK_MSG( !HasFlag(wxTR_MULTIPLE), wxTreeItemId(),
1305 wxT("this only works with single selection controls") );
1306
1307 return GetFocusedItem();
1308}
1309
1310wxTreeItemId wxTreeCtrl::GetFocusedItem() const
1311{
1312 return wxTreeItemId(TreeView_GetSelection(GetHwnd()));
1313}
1314
1315wxTreeItemId wxTreeCtrl::GetItemParent(const wxTreeItemId& item) const
1316{
1317 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1318
1319 HTREEITEM hItem;
1320
1321 if ( IS_VIRTUAL_ROOT(item) )
1322 {
1323 // no parent for the virtual root
1324 hItem = 0;
1325 }
1326 else // normal item
1327 {
1328 hItem = TreeView_GetParent(GetHwnd(), HITEM(item));
1329 if ( !hItem && HasFlag(wxTR_HIDE_ROOT) )
1330 {
1331 // the top level items should have the virtual root as their parent
1332 hItem = TVI_ROOT;
1333 }
1334 }
1335
1336 return wxTreeItemId(hItem);
1337}
1338
1339wxTreeItemId wxTreeCtrl::GetFirstChild(const wxTreeItemId& item,
1340 wxTreeItemIdValue& cookie) const
1341{
1342 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1343
1344 // remember the last child returned in 'cookie'
1345 cookie = TreeView_GetChild(GetHwnd(), HITEM(item));
1346
1347 return wxTreeItemId(cookie);
1348}
1349
1350wxTreeItemId wxTreeCtrl::GetNextChild(const wxTreeItemId& WXUNUSED(item),
1351 wxTreeItemIdValue& cookie) const
1352{
1353 wxTreeItemId fromCookie(cookie);
1354
1355 HTREEITEM hitem = HITEM(fromCookie);
1356
1357 hitem = TreeView_GetNextSibling(GetHwnd(), hitem);
1358
1359 wxTreeItemId item(hitem);
1360
1361 cookie = item.m_pItem;
1362
1363 return item;
1364}
1365
1366wxTreeItemId wxTreeCtrl::GetLastChild(const wxTreeItemId& item) const
1367{
1368 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1369
1370 // can this be done more efficiently?
1371 wxTreeItemIdValue cookie;
1372
1373 wxTreeItemId childLast,
1374 child = GetFirstChild(item, cookie);
1375 while ( child.IsOk() )
1376 {
1377 childLast = child;
1378 child = GetNextChild(item, cookie);
1379 }
1380
1381 return childLast;
1382}
1383
1384wxTreeItemId wxTreeCtrl::GetNextSibling(const wxTreeItemId& item) const
1385{
1386 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1387 return wxTreeItemId(TreeView_GetNextSibling(GetHwnd(), HITEM(item)));
1388}
1389
1390wxTreeItemId wxTreeCtrl::GetPrevSibling(const wxTreeItemId& item) const
1391{
1392 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1393 return wxTreeItemId(TreeView_GetPrevSibling(GetHwnd(), HITEM(item)));
1394}
1395
1396wxTreeItemId wxTreeCtrl::GetFirstVisibleItem() const
1397{
1398 return wxTreeItemId(TreeView_GetFirstVisible(GetHwnd()));
1399}
1400
1401wxTreeItemId wxTreeCtrl::GetNextVisible(const wxTreeItemId& item) const
1402{
1403 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1404 wxASSERT_MSG( IsVisible(item), wxT("The item you call GetNextVisible() for must be visible itself!"));
1405
1406 wxTreeItemId next(TreeView_GetNextVisible(GetHwnd(), HITEM(item)));
1407 if ( next.IsOk() && !IsVisible(next) )
1408 {
1409 // Win32 considers that any non-collapsed item is visible while we want
1410 // to return only really visible items
1411 next.Unset();
1412 }
1413
1414 return next;
1415}
1416
1417wxTreeItemId wxTreeCtrl::GetPrevVisible(const wxTreeItemId& item) const
1418{
1419 wxCHECK_MSG( item.IsOk(), wxTreeItemId(), wxT("invalid tree item") );
1420 wxASSERT_MSG( IsVisible(item), wxT("The item you call GetPrevVisible() for must be visible itself!"));
1421
1422 wxTreeItemId prev(TreeView_GetPrevVisible(GetHwnd(), HITEM(item)));
1423 if ( prev.IsOk() && !IsVisible(prev) )
1424 {
1425 // just as above, Win32 function will happily return the previous item
1426 // in the tree for the first visible item too
1427 prev.Unset();
1428 }
1429
1430 return prev;
1431}
1432
1433// ----------------------------------------------------------------------------
1434// multiple selections emulation
1435// ----------------------------------------------------------------------------
1436
1437size_t wxTreeCtrl::GetSelections(wxArrayTreeItemIds& selections) const
1438{
1439 TraverseSelections selector(this, selections);
1440
1441 return selector.GetCount();
1442}
1443
1444// ----------------------------------------------------------------------------
1445// Usual operations
1446// ----------------------------------------------------------------------------
1447
1448wxTreeItemId wxTreeCtrl::DoInsertAfter(const wxTreeItemId& parent,
1449 const wxTreeItemId& hInsertAfter,
1450 const wxString& text,
1451 int image, int selectedImage,
1452 wxTreeItemData *data)
1453{
1454 wxCHECK_MSG( parent.IsOk() || !TreeView_GetRoot(GetHwnd()),
1455 wxTreeItemId(),
1456 wxT("can't have more than one root in the tree") );
1457
1458 TV_INSERTSTRUCT tvIns;
1459 tvIns.hParent = HITEM(parent);
1460 tvIns.hInsertAfter = HITEM(hInsertAfter);
1461
1462 // this is how we insert the item as the first child: supply a NULL
1463 // hInsertAfter
1464 if ( !tvIns.hInsertAfter )
1465 {
1466 tvIns.hInsertAfter = TVI_FIRST;
1467 }
1468
1469 UINT mask = 0;
1470 if ( !text.empty() )
1471 {
1472 mask |= TVIF_TEXT;
1473 tvIns.item.pszText = (wxChar *)text.wx_str(); // cast is ok
1474 }
1475 else
1476 {
1477 tvIns.item.pszText = NULL;
1478 tvIns.item.cchTextMax = 0;
1479 }
1480
1481 // create the param which will store the other item parameters
1482 wxTreeItemParam *param = new wxTreeItemParam;
1483
1484 // we return the images on demand as they depend on whether the item is
1485 // expanded or collapsed too in our case
1486 mask |= TVIF_IMAGE | TVIF_SELECTEDIMAGE;
1487 tvIns.item.iImage = I_IMAGECALLBACK;
1488 tvIns.item.iSelectedImage = I_IMAGECALLBACK;
1489
1490 param->SetImage(image, wxTreeItemIcon_Normal);
1491 param->SetImage(selectedImage, wxTreeItemIcon_Selected);
1492
1493 mask |= TVIF_PARAM;
1494 tvIns.item.lParam = (LPARAM)param;
1495 tvIns.item.mask = mask;
1496
1497 // don't use the hack below for the children of hidden root: this results
1498 // in a crash inside comctl32.dll when we call TreeView_GetItemRect()
1499 const bool firstChild = !IsHiddenRoot(parent) &&
1500 !TreeView_GetChild(GetHwnd(), HITEM(parent));
1501
1502 HTREEITEM id = TreeView_InsertItem(GetHwnd(), &tvIns);
1503 if ( id == 0 )
1504 {
1505 wxLogLastError(wxT("TreeView_InsertItem"));
1506 }
1507
1508 // apparently some Windows versions (2000 and XP are reported to do this)
1509 // sometimes don't refresh the tree after adding the first child and so we
1510 // need this to make the "[+]" appear
1511 if ( firstChild )
1512 {
1513 TVGetItemRectParam param;
1514
1515 wxTreeView_GetItemRect(GetHwnd(), HITEM(parent), param, FALSE);
1516 ::InvalidateRect(GetHwnd(), &param.rect, FALSE);
1517 }
1518
1519 // associate the application tree item with Win32 tree item handle
1520 param->SetItem(id);
1521
1522 // setup wxTreeItemData
1523 if ( data != NULL )
1524 {
1525 param->SetData(data);
1526 data->SetId(id);
1527 }
1528
1529 return wxTreeItemId(id);
1530}
1531
1532wxTreeItemId wxTreeCtrl::AddRoot(const wxString& text,
1533 int image, int selectedImage,
1534 wxTreeItemData *data)
1535{
1536 if ( HasFlag(wxTR_HIDE_ROOT) )
1537 {
1538 wxASSERT_MSG( !m_pVirtualRoot, wxT("tree can have only a single root") );
1539
1540 // create a virtual root item, the parent for all the others
1541 wxTreeItemParam *param = new wxTreeItemParam;
1542 param->SetData(data);
1543
1544 m_pVirtualRoot = new wxVirtualNode(param);
1545
1546 return TVI_ROOT;
1547 }
1548
1549 return DoInsertAfter(wxTreeItemId(), wxTreeItemId(),
1550 text, image, selectedImage, data);
1551}
1552
1553wxTreeItemId wxTreeCtrl::DoInsertItem(const wxTreeItemId& parent,
1554 size_t index,
1555 const wxString& text,
1556 int image, int selectedImage,
1557 wxTreeItemData *data)
1558{
1559 wxTreeItemId idPrev;
1560 if ( index == (size_t)-1 )
1561 {
1562 // special value: append to the end
1563 idPrev = TVI_LAST;
1564 }
1565 else // find the item from index
1566 {
1567 wxTreeItemIdValue cookie;
1568 wxTreeItemId idCur = GetFirstChild(parent, cookie);
1569 while ( index != 0 && idCur.IsOk() )
1570 {
1571 index--;
1572
1573 idPrev = idCur;
1574 idCur = GetNextChild(parent, cookie);
1575 }
1576
1577 // assert, not check: if the index is invalid, we will append the item
1578 // to the end
1579 wxASSERT_MSG( index == 0, wxT("bad index in wxTreeCtrl::InsertItem") );
1580 }
1581
1582 return DoInsertAfter(parent, idPrev, text, image, selectedImage, data);
1583}
1584
1585void wxTreeCtrl::Delete(const wxTreeItemId& item)
1586{
1587 // unlock tree selections on vista, without this the
1588 // tree ctrl will eventually crash after item deletion
1589 TreeItemUnlocker unlock_all;
1590
1591 if ( HasFlag(wxTR_MULTIPLE) )
1592 {
1593 bool selected = IsSelected(item);
1594 wxTreeItemId next;
1595
1596 if ( selected )
1597 {
1598 next = TreeView_GetNextVisible(GetHwnd(), HITEM(item));
1599
1600 if ( !next.IsOk() )
1601 {
1602 next = TreeView_GetPrevVisible(GetHwnd(), HITEM(item));
1603 }
1604 }
1605
1606 {
1607 TempSetter set(m_changingSelection);
1608 if ( !TreeView_DeleteItem(GetHwnd(), HITEM(item)) )
1609 {
1610 wxLogLastError(wxT("TreeView_DeleteItem"));
1611 return;
1612 }
1613 }
1614
1615 if ( !selected )
1616 {
1617 return;
1618 }
1619
1620 if ( item == m_htSelStart )
1621 m_htSelStart.Unset();
1622
1623 if ( item == m_htClickedItem )
1624 m_htClickedItem.Unset();
1625
1626 if ( next.IsOk() )
1627 {
1628 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this, next);
1629
1630 if ( IsTreeEventAllowed(changingEvent) )
1631 {
1632 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED, this, next);
1633 (void)HandleTreeEvent(changedEvent);
1634 }
1635 else
1636 {
1637 DoUnselectItem(next);
1638 ClearFocusedItem();
1639 }
1640 }
1641 }
1642 else
1643 {
1644 if ( !TreeView_DeleteItem(GetHwnd(), HITEM(item)) )
1645 {
1646 wxLogLastError(wxT("TreeView_DeleteItem"));
1647 }
1648 }
1649}
1650
1651// delete all children (but don't delete the item itself)
1652void wxTreeCtrl::DeleteChildren(const wxTreeItemId& item)
1653{
1654 // unlock tree selections on vista for the duration of this call
1655 TreeItemUnlocker unlock_all;
1656
1657 wxTreeItemIdValue cookie;
1658
1659 wxArrayTreeItemIds children;
1660 wxTreeItemId child = GetFirstChild(item, cookie);
1661 while ( child.IsOk() )
1662 {
1663 children.Add(child);
1664
1665 child = GetNextChild(item, cookie);
1666 }
1667
1668 size_t nCount = children.Count();
1669 for ( size_t n = 0; n < nCount; n++ )
1670 {
1671 Delete(children[n]);
1672 }
1673}
1674
1675void wxTreeCtrl::DeleteAllItems()
1676{
1677 // unlock tree selections on vista for the duration of this call
1678 TreeItemUnlocker unlock_all;
1679
1680 // invalidate all the items we store as they're going to become invalid
1681 m_htSelStart =
1682 m_htClickedItem = wxTreeItemId();
1683
1684 // delete the "virtual" root item.
1685 if ( GET_VIRTUAL_ROOT() )
1686 {
1687 delete GET_VIRTUAL_ROOT();
1688 m_pVirtualRoot = NULL;
1689 }
1690
1691 // and all the real items
1692
1693 if ( !TreeView_DeleteAllItems(GetHwnd()) )
1694 {
1695 wxLogLastError(wxT("TreeView_DeleteAllItems"));
1696 }
1697}
1698
1699void wxTreeCtrl::DoExpand(const wxTreeItemId& item, int flag)
1700{
1701 wxASSERT_MSG( flag == TVE_COLLAPSE ||
1702 flag == (TVE_COLLAPSE | TVE_COLLAPSERESET) ||
1703 flag == TVE_EXPAND ||
1704 flag == TVE_TOGGLE,
1705 wxT("Unknown flag in wxTreeCtrl::DoExpand") );
1706
1707 // A hidden root can be neither expanded nor collapsed.
1708 wxCHECK_RET( !IsHiddenRoot(item),
1709 wxT("Can't expand/collapse hidden root node!") );
1710
1711 // TreeView_Expand doesn't send TVN_ITEMEXPAND(ING) messages, so we must
1712 // emulate them. This behaviour has changed slightly with comctl32.dll
1713 // v 4.70 - now it does send them but only the first time. To maintain
1714 // compatible behaviour and also in order to not have surprises with the
1715 // future versions, don't rely on this and still do everything ourselves.
1716 // To avoid that the messages be sent twice when the item is expanded for
1717 // the first time we must clear TVIS_EXPANDEDONCE style manually.
1718
1719 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_EXPANDEDONCE);
1720 tvItem.state = 0;
1721 DoSetItem(&tvItem);
1722
1723 if ( IsExpanded(item) )
1724 {
1725 wxTreeEvent event(wxEVT_COMMAND_TREE_ITEM_COLLAPSING,
1726 this, wxTreeItemId(item));
1727
1728 if ( !IsTreeEventAllowed(event) )
1729 return;
1730 }
1731
1732 if ( TreeView_Expand(GetHwnd(), HITEM(item), flag) )
1733 {
1734 if ( IsExpanded(item) )
1735 return;
1736
1737 wxTreeEvent event(wxEVT_COMMAND_TREE_ITEM_COLLAPSED, this, item);
1738 (void)HandleTreeEvent(event);
1739 }
1740 //else: change didn't took place, so do nothing at all
1741}
1742
1743void wxTreeCtrl::Expand(const wxTreeItemId& item)
1744{
1745 DoExpand(item, TVE_EXPAND);
1746}
1747
1748void wxTreeCtrl::Collapse(const wxTreeItemId& item)
1749{
1750 DoExpand(item, TVE_COLLAPSE);
1751}
1752
1753void wxTreeCtrl::CollapseAndReset(const wxTreeItemId& item)
1754{
1755 DoExpand(item, TVE_COLLAPSE | TVE_COLLAPSERESET);
1756}
1757
1758void wxTreeCtrl::Toggle(const wxTreeItemId& item)
1759{
1760 DoExpand(item, TVE_TOGGLE);
1761}
1762
1763void wxTreeCtrl::Unselect()
1764{
1765 wxASSERT_MSG( !HasFlag(wxTR_MULTIPLE),
1766 wxT("doesn't make sense, may be you want UnselectAll()?") );
1767
1768 // the current focus
1769 HTREEITEM htFocus = (HTREEITEM)TreeView_GetSelection(GetHwnd());
1770
1771 if ( !htFocus )
1772 {
1773 return;
1774 }
1775
1776 if ( HasFlag(wxTR_MULTIPLE) )
1777 {
1778 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
1779 this, wxTreeItemId());
1780 changingEvent.m_itemOld = htFocus;
1781
1782 if ( IsTreeEventAllowed(changingEvent) )
1783 {
1784 ClearFocusedItem();
1785
1786 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
1787 this, wxTreeItemId());
1788 changedEvent.m_itemOld = htFocus;
1789 (void)HandleTreeEvent(changedEvent);
1790 }
1791 }
1792 else
1793 {
1794 ClearFocusedItem();
1795 }
1796}
1797
1798void wxTreeCtrl::DoUnselectAll()
1799{
1800 wxArrayTreeItemIds selections;
1801 size_t count = GetSelections(selections);
1802
1803 for ( size_t n = 0; n < count; n++ )
1804 {
1805 DoUnselectItem(selections[n]);
1806 }
1807
1808 m_htSelStart.Unset();
1809}
1810
1811void wxTreeCtrl::UnselectAll()
1812{
1813 if ( HasFlag(wxTR_MULTIPLE) )
1814 {
1815 HTREEITEM htFocus = (HTREEITEM)TreeView_GetSelection(GetHwnd());
1816 if ( !htFocus ) return;
1817
1818 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this);
1819 changingEvent.m_itemOld = htFocus;
1820
1821 if ( IsTreeEventAllowed(changingEvent) )
1822 {
1823 DoUnselectAll();
1824
1825 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED, this);
1826 changedEvent.m_itemOld = htFocus;
1827 (void)HandleTreeEvent(changedEvent);
1828 }
1829 }
1830 else
1831 {
1832 Unselect();
1833 }
1834}
1835
1836void wxTreeCtrl::DoSelectChildren(const wxTreeItemId& parent)
1837{
1838 DoUnselectAll();
1839
1840 wxTreeItemIdValue cookie;
1841 wxTreeItemId child = GetFirstChild(parent, cookie);
1842 while ( child.IsOk() )
1843 {
1844 DoSelectItem(child, true);
1845 child = GetNextChild(child, cookie);
1846 }
1847}
1848
1849void wxTreeCtrl::SelectChildren(const wxTreeItemId& parent)
1850{
1851 wxCHECK_RET( HasFlag(wxTR_MULTIPLE),
1852 "this only works with multiple selection controls" );
1853
1854 HTREEITEM htFocus = (HTREEITEM)TreeView_GetSelection(GetHwnd());
1855
1856 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this);
1857 changingEvent.m_itemOld = htFocus;
1858
1859 if ( IsTreeEventAllowed(changingEvent) )
1860 {
1861 DoSelectChildren(parent);
1862
1863 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED, this);
1864 changedEvent.m_itemOld = htFocus;
1865 (void)HandleTreeEvent(changedEvent);
1866 }
1867}
1868
1869void wxTreeCtrl::DoSelectItem(const wxTreeItemId& item, bool select)
1870{
1871 TempSetter set(m_changingSelection);
1872
1873 ::SelectItem(GetHwnd(), HITEM(item), select);
1874}
1875
1876void wxTreeCtrl::SelectItem(const wxTreeItemId& item, bool select)
1877{
1878 wxCHECK_RET( !IsHiddenRoot(item), wxT("can't select hidden root item") );
1879
1880 if ( select == IsSelected(item) )
1881 {
1882 // nothing to do, the item is already in the requested state
1883 return;
1884 }
1885
1886 if ( HasFlag(wxTR_MULTIPLE) )
1887 {
1888 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this, item);
1889
1890 if ( IsTreeEventAllowed(changingEvent) )
1891 {
1892 HTREEITEM htFocus = (HTREEITEM)TreeView_GetSelection(GetHwnd());
1893 DoSelectItem(item, select);
1894
1895 if ( !htFocus )
1896 {
1897 SetFocusedItem(item);
1898 }
1899
1900 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
1901 this, item);
1902 (void)HandleTreeEvent(changedEvent);
1903 }
1904 }
1905 else // single selection
1906 {
1907 wxTreeItemId itemOld, itemNew;
1908 if ( select )
1909 {
1910 itemOld = GetSelection();
1911 itemNew = item;
1912 }
1913 else // deselecting the currently selected item
1914 {
1915 itemOld = item;
1916 // leave itemNew invalid
1917 }
1918
1919 // Recent versions of comctl32.dll send TVN_SELCHANG{ED,ING} events
1920 // when we call TreeView_SelectItem() but apparently some old ones did
1921 // not so send the events ourselves and ignore those generated by
1922 // TreeView_SelectItem() if m_changingSelection is set.
1923 wxTreeEvent
1924 changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this, itemNew);
1925 changingEvent.SetOldItem(itemOld);
1926
1927 if ( IsTreeEventAllowed(changingEvent) )
1928 {
1929 TempSetter set(m_changingSelection);
1930
1931 if ( !TreeView_SelectItem(GetHwnd(), HITEM(itemNew)) )
1932 {
1933 wxLogLastError(wxT("TreeView_SelectItem"));
1934 }
1935 else // ok
1936 {
1937 ::SetFocus(GetHwnd(), HITEM(item));
1938
1939 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
1940 this, itemNew);
1941 changedEvent.SetOldItem(itemOld);
1942 (void)HandleTreeEvent(changedEvent);
1943 }
1944 }
1945 //else: program vetoed the change
1946 }
1947}
1948
1949void wxTreeCtrl::EnsureVisible(const wxTreeItemId& item)
1950{
1951 wxCHECK_RET( !IsHiddenRoot(item), wxT("can't show hidden root item") );
1952
1953 // no error return
1954 TreeView_EnsureVisible(GetHwnd(), HITEM(item));
1955}
1956
1957void wxTreeCtrl::ScrollTo(const wxTreeItemId& item)
1958{
1959 if ( !TreeView_SelectSetFirstVisible(GetHwnd(), HITEM(item)) )
1960 {
1961 wxLogLastError(wxT("TreeView_SelectSetFirstVisible"));
1962 }
1963}
1964
1965wxTextCtrl *wxTreeCtrl::GetEditControl() const
1966{
1967 return m_textCtrl;
1968}
1969
1970void wxTreeCtrl::DeleteTextCtrl()
1971{
1972 if ( m_textCtrl )
1973 {
1974 // the HWND corresponding to this control is deleted by the tree
1975 // control itself and we don't know when exactly this happens, so check
1976 // if the window still exists before calling UnsubclassWin()
1977 if ( !::IsWindow(GetHwndOf(m_textCtrl)) )
1978 {
1979 m_textCtrl->SetHWND(0);
1980 }
1981
1982 m_textCtrl->UnsubclassWin();
1983 m_textCtrl->SetHWND(0);
1984 wxDELETE(m_textCtrl);
1985
1986 m_idEdited.Unset();
1987 }
1988}
1989
1990wxTextCtrl *wxTreeCtrl::EditLabel(const wxTreeItemId& item,
1991 wxClassInfo *textControlClass)
1992{
1993 wxASSERT( textControlClass->IsKindOf(CLASSINFO(wxTextCtrl)) );
1994
1995 DeleteTextCtrl();
1996
1997 m_idEdited = item;
1998 m_textCtrl = (wxTextCtrl *)textControlClass->CreateObject();
1999 HWND hWnd = (HWND) TreeView_EditLabel(GetHwnd(), HITEM(item));
2000
2001 // this is not an error - the TVN_BEGINLABELEDIT handler might have
2002 // returned false
2003 if ( !hWnd )
2004 {
2005 wxDELETE(m_textCtrl);
2006 return NULL;
2007 }
2008
2009 // textctrl is subclassed in MSWOnNotify
2010 return m_textCtrl;
2011}
2012
2013// End label editing, optionally cancelling the edit
2014void wxTreeCtrl::DoEndEditLabel(bool discardChanges)
2015{
2016 TreeView_EndEditLabelNow(GetHwnd(), discardChanges);
2017
2018 DeleteTextCtrl();
2019}
2020
2021wxTreeItemId wxTreeCtrl::DoTreeHitTest(const wxPoint& point, int& flags) const
2022{
2023 TV_HITTESTINFO hitTestInfo;
2024 hitTestInfo.pt.x = (int)point.x;
2025 hitTestInfo.pt.y = (int)point.y;
2026
2027 (void) TreeView_HitTest(GetHwnd(), &hitTestInfo);
2028
2029 flags = 0;
2030
2031 // avoid repetition
2032 #define TRANSLATE_FLAG(flag) if ( hitTestInfo.flags & TVHT_##flag ) \
2033 flags |= wxTREE_HITTEST_##flag
2034
2035 TRANSLATE_FLAG(ABOVE);
2036 TRANSLATE_FLAG(BELOW);
2037 TRANSLATE_FLAG(NOWHERE);
2038 TRANSLATE_FLAG(ONITEMBUTTON);
2039 TRANSLATE_FLAG(ONITEMICON);
2040 TRANSLATE_FLAG(ONITEMINDENT);
2041 TRANSLATE_FLAG(ONITEMLABEL);
2042 TRANSLATE_FLAG(ONITEMRIGHT);
2043 TRANSLATE_FLAG(ONITEMSTATEICON);
2044 TRANSLATE_FLAG(TOLEFT);
2045 TRANSLATE_FLAG(TORIGHT);
2046
2047 #undef TRANSLATE_FLAG
2048
2049 return wxTreeItemId(hitTestInfo.hItem);
2050}
2051
2052bool wxTreeCtrl::GetBoundingRect(const wxTreeItemId& item,
2053 wxRect& rect,
2054 bool textOnly) const
2055{
2056 // Virtual root items have no bounding rectangle
2057 if ( IS_VIRTUAL_ROOT(item) )
2058 {
2059 return false;
2060 }
2061
2062 TVGetItemRectParam param;
2063
2064 if ( wxTreeView_GetItemRect(GetHwnd(), HITEM(item), param, textOnly) )
2065 {
2066 rect = wxRect(wxPoint(param.rect.left, param.rect.top),
2067 wxPoint(param.rect.right, param.rect.bottom));
2068
2069 return true;
2070 }
2071 else
2072 {
2073 // couldn't retrieve rect: for example, item isn't visible
2074 return false;
2075 }
2076}
2077
2078void wxTreeCtrl::ClearFocusedItem()
2079{
2080 TempSetter set(m_changingSelection);
2081
2082 if ( !TreeView_SelectItem(GetHwnd(), 0) )
2083 {
2084 wxLogLastError(wxT("TreeView_SelectItem"));
2085 }
2086}
2087
2088void wxTreeCtrl::SetFocusedItem(const wxTreeItemId& item)
2089{
2090 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2091
2092 TempSetter set(m_changingSelection);
2093
2094 ::SetFocus(GetHwnd(), HITEM(item));
2095}
2096
2097void wxTreeCtrl::DoUnselectItem(const wxTreeItemId& item)
2098{
2099 TempSetter set(m_changingSelection);
2100
2101 ::UnselectItem(GetHwnd(), HITEM(item));
2102}
2103
2104void wxTreeCtrl::DoToggleItemSelection(const wxTreeItemId& item)
2105{
2106 TempSetter set(m_changingSelection);
2107
2108 ::ToggleItemSelection(GetHwnd(), HITEM(item));
2109}
2110
2111// ----------------------------------------------------------------------------
2112// sorting stuff
2113// ----------------------------------------------------------------------------
2114
2115// this is just a tiny namespace which is friend to wxTreeCtrl and so can use
2116// functions such as IsDataIndirect()
2117class wxTreeSortHelper
2118{
2119public:
2120 static int CALLBACK Compare(LPARAM data1, LPARAM data2, LPARAM tree);
2121
2122private:
2123 static wxTreeItemId GetIdFromData(LPARAM lParam)
2124 {
2125 return ((wxTreeItemParam*)lParam)->GetItem();
2126 }
2127};
2128
2129int CALLBACK wxTreeSortHelper::Compare(LPARAM pItem1,
2130 LPARAM pItem2,
2131 LPARAM htree)
2132{
2133 wxCHECK_MSG( pItem1 && pItem2, 0,
2134 wxT("sorting tree without data doesn't make sense") );
2135
2136 wxTreeCtrl *tree = (wxTreeCtrl *)htree;
2137
2138 return tree->OnCompareItems(GetIdFromData(pItem1),
2139 GetIdFromData(pItem2));
2140}
2141
2142void wxTreeCtrl::SortChildren(const wxTreeItemId& item)
2143{
2144 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
2145
2146 // rely on the fact that TreeView_SortChildren does the same thing as our
2147 // default behaviour, i.e. sorts items alphabetically and so call it
2148 // directly if we're not in derived class (much more efficient!)
2149 // RN: Note that if you find you're code doesn't sort as expected this
2150 // may be why as if you don't use the DECLARE_CLASS/IMPLEMENT_CLASS
2151 // combo for your derived wxTreeCtrl if will sort without
2152 // OnCompareItems
2153 if ( GetClassInfo() == CLASSINFO(wxTreeCtrl) )
2154 {
2155 TreeView_SortChildren(GetHwnd(), HITEM(item), 0);
2156 }
2157 else
2158 {
2159 TV_SORTCB tvSort;
2160 tvSort.hParent = HITEM(item);
2161 tvSort.lpfnCompare = wxTreeSortHelper::Compare;
2162 tvSort.lParam = (LPARAM)this;
2163 TreeView_SortChildrenCB(GetHwnd(), &tvSort, 0 /* reserved */);
2164 }
2165}
2166
2167// ----------------------------------------------------------------------------
2168// implementation
2169// ----------------------------------------------------------------------------
2170
2171bool wxTreeCtrl::MSWShouldPreProcessMessage(WXMSG* msg)
2172{
2173 if ( msg->message == WM_KEYDOWN )
2174 {
2175 // Only eat VK_RETURN if not being used by the application in
2176 // conjunction with modifiers
2177 if ( (msg->wParam == VK_RETURN) && !wxIsAnyModifierDown() )
2178 {
2179 // we need VK_RETURN to generate wxEVT_COMMAND_TREE_ITEM_ACTIVATED
2180 return false;
2181 }
2182 }
2183
2184 return wxTreeCtrlBase::MSWShouldPreProcessMessage(msg);
2185}
2186
2187bool wxTreeCtrl::MSWCommand(WXUINT cmd, WXWORD id_)
2188{
2189 const int id = (signed short)id_;
2190
2191 if ( cmd == EN_UPDATE )
2192 {
2193 wxCommandEvent event(wxEVT_COMMAND_TEXT_UPDATED, id);
2194 event.SetEventObject( this );
2195 ProcessCommand(event);
2196 }
2197 else if ( cmd == EN_KILLFOCUS )
2198 {
2199 wxCommandEvent event(wxEVT_KILL_FOCUS, id);
2200 event.SetEventObject( this );
2201 ProcessCommand(event);
2202 }
2203 else
2204 {
2205 // nothing done
2206 return false;
2207 }
2208
2209 // command processed
2210 return true;
2211}
2212
2213bool wxTreeCtrl::MSWIsOnItem(unsigned flags) const
2214{
2215 unsigned mask = TVHT_ONITEM;
2216 if ( HasFlag(wxTR_FULL_ROW_HIGHLIGHT) )
2217 mask |= TVHT_ONITEMINDENT | TVHT_ONITEMRIGHT;
2218
2219 return (flags & mask) != 0;
2220}
2221
2222bool wxTreeCtrl::MSWHandleSelectionKey(unsigned vkey)
2223{
2224 const bool bCtrl = wxIsCtrlDown();
2225 const bool bShift = wxIsShiftDown();
2226 const HTREEITEM htSel = (HTREEITEM)TreeView_GetSelection(GetHwnd());
2227
2228 switch ( vkey )
2229 {
2230 case VK_RETURN:
2231 case VK_SPACE:
2232 if ( !htSel )
2233 break;
2234
2235 if ( vkey != VK_RETURN && bCtrl )
2236 {
2237 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2238 this, htSel);
2239 changingEvent.m_itemOld = htSel;
2240
2241 if ( IsTreeEventAllowed(changingEvent) )
2242 {
2243 DoToggleItemSelection(wxTreeItemId(htSel));
2244
2245 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2246 this, htSel);
2247 changedEvent.m_itemOld = htSel;
2248 (void)HandleTreeEvent(changedEvent);
2249 }
2250 }
2251 else
2252 {
2253 wxArrayTreeItemIds selections;
2254 size_t count = GetSelections(selections);
2255
2256 if ( count != 1 || HITEM(selections[0]) != htSel )
2257 {
2258 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2259 this, htSel);
2260 changingEvent.m_itemOld = htSel;
2261
2262 if ( IsTreeEventAllowed(changingEvent) )
2263 {
2264 DoUnselectAll();
2265 DoSelectItem(wxTreeItemId(htSel));
2266
2267 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2268 this, htSel);
2269 changedEvent.m_itemOld = htSel;
2270 (void)HandleTreeEvent(changedEvent);
2271 }
2272 }
2273 }
2274 break;
2275
2276 case VK_UP:
2277 case VK_DOWN:
2278 if ( !bCtrl && !bShift )
2279 {
2280 wxArrayTreeItemIds selections;
2281 wxTreeItemId next;
2282
2283 if ( htSel )
2284 {
2285 next = vkey == VK_UP
2286 ? TreeView_GetPrevVisible(GetHwnd(), htSel)
2287 : TreeView_GetNextVisible(GetHwnd(), htSel);
2288 }
2289 else
2290 {
2291 next = GetRootItem();
2292
2293 if ( IsHiddenRoot(next) )
2294 next = TreeView_GetChild(GetHwnd(), HITEM(next));
2295 }
2296
2297 if ( !next.IsOk() )
2298 {
2299 break;
2300 }
2301
2302 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2303 this, next);
2304 changingEvent.m_itemOld = htSel;
2305
2306 if ( IsTreeEventAllowed(changingEvent) )
2307 {
2308 DoUnselectAll();
2309 DoSelectItem(next);
2310 SetFocusedItem(next);
2311
2312 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2313 this, next);
2314 changedEvent.m_itemOld = htSel;
2315 (void)HandleTreeEvent(changedEvent);
2316 }
2317 }
2318 else if ( htSel )
2319 {
2320 wxTreeItemId next = vkey == VK_UP
2321 ? TreeView_GetPrevVisible(GetHwnd(), htSel)
2322 : TreeView_GetNextVisible(GetHwnd(), htSel);
2323
2324 if ( !next.IsOk() )
2325 {
2326 break;
2327 }
2328
2329 if ( !m_htSelStart )
2330 {
2331 m_htSelStart = htSel;
2332 }
2333
2334 if ( bShift && SelectRange(GetHwnd(), HITEM(m_htSelStart), HITEM(next),
2335 SR_UNSELECT_OTHERS | SR_SIMULATE) )
2336 {
2337 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this, next);
2338 changingEvent.m_itemOld = htSel;
2339
2340 if ( IsTreeEventAllowed(changingEvent) )
2341 {
2342 SelectRange(GetHwnd(), HITEM(m_htSelStart), HITEM(next),
2343 SR_UNSELECT_OTHERS);
2344
2345 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED, this, next);
2346 changedEvent.m_itemOld = htSel;
2347 (void)HandleTreeEvent(changedEvent);
2348 }
2349 }
2350
2351 SetFocusedItem(next);
2352 }
2353 break;
2354
2355 case VK_LEFT:
2356 if ( HasChildren(htSel) && IsExpanded(htSel) )
2357 {
2358 Collapse(htSel);
2359 }
2360 else
2361 {
2362 wxTreeItemId next = GetItemParent(htSel);
2363
2364 if ( next.IsOk() && !IsHiddenRoot(next) )
2365 {
2366 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2367 this, next);
2368 changingEvent.m_itemOld = htSel;
2369
2370 if ( IsTreeEventAllowed(changingEvent) )
2371 {
2372 DoUnselectAll();
2373 DoSelectItem(next);
2374 SetFocusedItem(next);
2375
2376 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2377 this, next);
2378 changedEvent.m_itemOld = htSel;
2379 (void)HandleTreeEvent(changedEvent);
2380 }
2381 }
2382 }
2383 break;
2384
2385 case VK_RIGHT:
2386 if ( !IsVisible(htSel) )
2387 {
2388 EnsureVisible(htSel);
2389 }
2390
2391 if ( !HasChildren(htSel) )
2392 break;
2393
2394 if ( !IsExpanded(htSel) )
2395 {
2396 Expand(htSel);
2397 }
2398 else
2399 {
2400 wxTreeItemId next = TreeView_GetChild(GetHwnd(), htSel);
2401
2402 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING, this, next);
2403 changingEvent.m_itemOld = htSel;
2404
2405 if ( IsTreeEventAllowed(changingEvent) )
2406 {
2407 DoUnselectAll();
2408 DoSelectItem(next);
2409 SetFocusedItem(next);
2410
2411 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED, this, next);
2412 changedEvent.m_itemOld = htSel;
2413 (void)HandleTreeEvent(changedEvent);
2414 }
2415 }
2416 break;
2417
2418 case VK_HOME:
2419 case VK_END:
2420 {
2421 wxTreeItemId next = GetRootItem();
2422
2423 if ( IsHiddenRoot(next) )
2424 {
2425 next = TreeView_GetChild(GetHwnd(), HITEM(next));
2426 }
2427
2428 if ( !next.IsOk() )
2429 break;
2430
2431 if ( vkey == VK_END )
2432 {
2433 for ( ;; )
2434 {
2435 wxTreeItemId nextTemp = TreeView_GetNextVisible(
2436 GetHwnd(), HITEM(next));
2437
2438 if ( !nextTemp.IsOk() )
2439 break;
2440
2441 next = nextTemp;
2442 }
2443 }
2444
2445 if ( htSel == HITEM(next) )
2446 break;
2447
2448 if ( bShift )
2449 {
2450 if ( !m_htSelStart )
2451 {
2452 m_htSelStart = htSel;
2453 }
2454
2455 if ( SelectRange(GetHwnd(),
2456 HITEM(m_htSelStart), HITEM(next),
2457 SR_UNSELECT_OTHERS | SR_SIMULATE) )
2458 {
2459 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2460 this, next);
2461 changingEvent.m_itemOld = htSel;
2462
2463 if ( IsTreeEventAllowed(changingEvent) )
2464 {
2465 SelectRange(GetHwnd(),
2466 HITEM(m_htSelStart), HITEM(next),
2467 SR_UNSELECT_OTHERS);
2468 SetFocusedItem(next);
2469
2470 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2471 this, next);
2472 changedEvent.m_itemOld = htSel;
2473 (void)HandleTreeEvent(changedEvent);
2474 }
2475 }
2476 }
2477 else // no Shift
2478 {
2479 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2480 this, next);
2481 changingEvent.m_itemOld = htSel;
2482
2483 if ( IsTreeEventAllowed(changingEvent) )
2484 {
2485 DoUnselectAll();
2486 DoSelectItem(next);
2487 SetFocusedItem(next);
2488
2489 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2490 this, next);
2491 changedEvent.m_itemOld = htSel;
2492 (void)HandleTreeEvent(changedEvent);
2493 }
2494 }
2495 }
2496 break;
2497
2498 case VK_PRIOR:
2499 case VK_NEXT:
2500 if ( bCtrl )
2501 {
2502 wxTreeItemId firstVisible = GetFirstVisibleItem();
2503 size_t visibleCount = TreeView_GetVisibleCount(GetHwnd());
2504 wxTreeItemId nextAdjacent = (vkey == VK_PRIOR) ?
2505 TreeView_GetPrevVisible(GetHwnd(), HITEM(firstVisible)) :
2506 TreeView_GetNextVisible(GetHwnd(), HITEM(firstVisible));
2507
2508 if ( !nextAdjacent )
2509 {
2510 break;
2511 }
2512
2513 wxTreeItemId nextStart = firstVisible;
2514
2515 for ( size_t n = 1; n < visibleCount; n++ )
2516 {
2517 wxTreeItemId nextTemp = (vkey == VK_PRIOR) ?
2518 TreeView_GetPrevVisible(GetHwnd(), HITEM(nextStart)) :
2519 TreeView_GetNextVisible(GetHwnd(), HITEM(nextStart));
2520
2521 if ( nextTemp.IsOk() )
2522 {
2523 nextStart = nextTemp;
2524 }
2525 else
2526 {
2527 break;
2528 }
2529 }
2530
2531 EnsureVisible(nextStart);
2532
2533 if ( vkey == VK_NEXT )
2534 {
2535 wxTreeItemId nextEnd = nextStart;
2536
2537 for ( size_t n = 1; n < visibleCount; n++ )
2538 {
2539 wxTreeItemId nextTemp =
2540 TreeView_GetNextVisible(GetHwnd(), HITEM(nextEnd));
2541
2542 if ( nextTemp.IsOk() )
2543 {
2544 nextEnd = nextTemp;
2545 }
2546 else
2547 {
2548 break;
2549 }
2550 }
2551
2552 EnsureVisible(nextEnd);
2553 }
2554 }
2555 else // no Ctrl
2556 {
2557 size_t visibleCount = TreeView_GetVisibleCount(GetHwnd());
2558 wxTreeItemId nextAdjacent = (vkey == VK_PRIOR) ?
2559 TreeView_GetPrevVisible(GetHwnd(), htSel) :
2560 TreeView_GetNextVisible(GetHwnd(), htSel);
2561
2562 if ( !nextAdjacent )
2563 {
2564 break;
2565 }
2566
2567 wxTreeItemId next(htSel);
2568
2569 for ( size_t n = 1; n < visibleCount; n++ )
2570 {
2571 wxTreeItemId nextTemp = vkey == VK_PRIOR ?
2572 TreeView_GetPrevVisible(GetHwnd(), HITEM(next)) :
2573 TreeView_GetNextVisible(GetHwnd(), HITEM(next));
2574
2575 if ( !nextTemp.IsOk() )
2576 break;
2577
2578 next = nextTemp;
2579 }
2580
2581 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2582 this, next);
2583 changingEvent.m_itemOld = htSel;
2584
2585 if ( IsTreeEventAllowed(changingEvent) )
2586 {
2587 DoUnselectAll();
2588 m_htSelStart.Unset();
2589 DoSelectItem(next);
2590 SetFocusedItem(next);
2591
2592 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2593 this, next);
2594 changedEvent.m_itemOld = htSel;
2595 (void)HandleTreeEvent(changedEvent);
2596 }
2597 }
2598 break;
2599
2600 default:
2601 return false;
2602 }
2603
2604 return true;
2605}
2606
2607bool wxTreeCtrl::MSWHandleTreeKeyDownEvent(WXWPARAM wParam, WXLPARAM lParam)
2608{
2609 wxTreeEvent keyEvent(wxEVT_COMMAND_TREE_KEY_DOWN, this);
2610 keyEvent.m_evtKey = CreateKeyEvent(wxEVT_KEY_DOWN, wParam, lParam);
2611
2612 bool processed = HandleTreeEvent(keyEvent);
2613
2614 // generate a separate event for Space/Return
2615 if ( !wxIsCtrlDown() && !wxIsShiftDown() && !wxIsAltDown() &&
2616 ((wParam == VK_SPACE) || (wParam == VK_RETURN)) )
2617 {
2618 const HTREEITEM htSel = (HTREEITEM)TreeView_GetSelection(GetHwnd());
2619 if ( htSel )
2620 {
2621 wxTreeEvent activatedEvent(wxEVT_COMMAND_TREE_ITEM_ACTIVATED,
2622 this, htSel);
2623 (void)HandleTreeEvent(activatedEvent);
2624 }
2625 }
2626
2627 return processed;
2628}
2629
2630// we hook into WndProc to process WM_MOUSEMOVE/WM_BUTTONUP messages - as we
2631// only do it during dragging, minimize wxWin overhead (this is important for
2632// WM_MOUSEMOVE as they're a lot of them) by catching Windows messages directly
2633// instead of passing by wxWin events
2634WXLRESULT
2635wxTreeCtrl::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
2636{
2637 bool processed = false;
2638 WXLRESULT rc = 0;
2639 bool isMultiple = HasFlag(wxTR_MULTIPLE);
2640
2641 if ( nMsg == WM_CONTEXTMENU )
2642 {
2643 int x = GET_X_LPARAM(lParam),
2644 y = GET_Y_LPARAM(lParam);
2645
2646 // the item for which the menu should be shown
2647 wxTreeItemId item;
2648
2649 // the position where the menu should be shown in client coordinates
2650 // (so that it can be passed directly to PopupMenu())
2651 wxPoint pt;
2652
2653 if ( x == -1 || y == -1 )
2654 {
2655 // this means that the event was generated from keyboard (e.g. with
2656 // Shift-F10 or special Windows menu key)
2657 //
2658 // use the Explorer standard of putting the menu at the left edge
2659 // of the text, in the vertical middle of the text
2660 item = wxTreeItemId(TreeView_GetSelection(GetHwnd()));
2661 if ( item.IsOk() )
2662 {
2663 // Use the bounding rectangle of only the text part
2664 wxRect rect;
2665 GetBoundingRect(item, rect, true);
2666 pt = wxPoint(rect.GetX(), rect.GetY() + rect.GetHeight() / 2);
2667 }
2668 }
2669 else // event from mouse, use mouse position
2670 {
2671 pt = ScreenToClient(wxPoint(x, y));
2672
2673 TV_HITTESTINFO tvhti;
2674 tvhti.pt.x = pt.x;
2675 tvhti.pt.y = pt.y;
2676
2677 if ( TreeView_HitTest(GetHwnd(), &tvhti) )
2678 item = wxTreeItemId(tvhti.hItem);
2679 }
2680
2681 // create the event
2682 if ( item.IsOk() )
2683 {
2684 wxTreeEvent event(wxEVT_COMMAND_TREE_ITEM_MENU, this, item);
2685
2686 event.m_pointDrag = pt;
2687
2688 if ( HandleTreeEvent(event) )
2689 processed = true;
2690 //else: continue with generating wxEVT_CONTEXT_MENU in base class code
2691 }
2692 }
2693 else if ( (nMsg >= WM_MOUSEFIRST) && (nMsg <= WM_MOUSELAST) )
2694 {
2695 // we only process mouse messages here and these parameters have the
2696 // same meaning for all of them
2697 int x = GET_X_LPARAM(lParam),
2698 y = GET_Y_LPARAM(lParam);
2699
2700 TV_HITTESTINFO tvht;
2701 tvht.pt.x = x;
2702 tvht.pt.y = y;
2703
2704 HTREEITEM htOldItem = TreeView_GetSelection(GetHwnd());
2705 HTREEITEM htItem = TreeView_HitTest(GetHwnd(), &tvht);
2706
2707 switch ( nMsg )
2708 {
2709 case WM_LBUTTONDOWN:
2710 if ( !isMultiple )
2711 break;
2712
2713 m_htClickedItem.Unset();
2714
2715 if ( !MSWIsOnItem(tvht.flags) )
2716 {
2717 if ( tvht.flags & TVHT_ONITEMBUTTON )
2718 {
2719 // either it's going to be handled by user code or
2720 // we're going to use it ourselves to toggle the
2721 // branch, in either case don't pass it to the base
2722 // class which would generate another mouse click event
2723 // for it even though it's already handled here
2724 processed = true;
2725 SetFocus();
2726
2727 if ( !HandleMouseEvent(nMsg, x, y, wParam) )
2728 {
2729 if ( !IsExpanded(htItem) )
2730 {
2731 Expand(htItem);
2732 }
2733 else
2734 {
2735 Collapse(htItem);
2736 }
2737 }
2738 }
2739
2740 m_focusLost = false;
2741 break;
2742 }
2743
2744 processed = true;
2745 SetFocus();
2746 m_htClickedItem = (WXHTREEITEM) htItem;
2747 m_ptClick = wxPoint(x, y);
2748
2749 if ( wParam & MK_CONTROL )
2750 {
2751 if ( HandleMouseEvent(nMsg, x, y, wParam) )
2752 {
2753 m_htClickedItem.Unset();
2754 break;
2755 }
2756
2757 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2758 this, htItem);
2759 changingEvent.m_itemOld = htOldItem;
2760
2761 if ( IsTreeEventAllowed(changingEvent) )
2762 {
2763 // toggle selected state
2764 DoToggleItemSelection(wxTreeItemId(htItem));
2765
2766 SetFocusedItem(wxTreeItemId(htItem));
2767
2768 // reset on any click without Shift
2769 m_htSelStart.Unset();
2770
2771 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2772 this, htItem);
2773 changedEvent.m_itemOld = htOldItem;
2774 (void)HandleTreeEvent(changedEvent);
2775 }
2776 }
2777 else if ( wParam & MK_SHIFT )
2778 {
2779 if ( HandleMouseEvent(nMsg, x, y, wParam) )
2780 {
2781 m_htClickedItem.Unset();
2782 break;
2783 }
2784
2785 int srFlags = 0;
2786 bool willChange = true;
2787
2788 if ( !(wParam & MK_CONTROL) )
2789 {
2790 srFlags |= SR_UNSELECT_OTHERS;
2791 }
2792
2793 if ( !m_htSelStart )
2794 {
2795 // take the focused item
2796 m_htSelStart = htOldItem;
2797 }
2798 else
2799 {
2800 willChange = SelectRange(GetHwnd(), HITEM(m_htSelStart),
2801 htItem, srFlags | SR_SIMULATE);
2802 }
2803
2804 if ( willChange )
2805 {
2806 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2807 this, htItem);
2808 changingEvent.m_itemOld = htOldItem;
2809
2810 if ( IsTreeEventAllowed(changingEvent) )
2811 {
2812 // this selects all items between the starting one
2813 // and the current
2814 if ( m_htSelStart )
2815 {
2816 SelectRange(GetHwnd(), HITEM(m_htSelStart),
2817 htItem, srFlags);
2818 }
2819 else
2820 {
2821 DoSelectItem(wxTreeItemId(htItem));
2822 }
2823
2824 SetFocusedItem(wxTreeItemId(htItem));
2825
2826 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2827 this, htItem);
2828 changedEvent.m_itemOld = htOldItem;
2829 (void)HandleTreeEvent(changedEvent);
2830 }
2831 }
2832 }
2833 else // normal click
2834 {
2835 // avoid doing anything if we click on the only
2836 // currently selected item
2837
2838 wxArrayTreeItemIds selections;
2839 size_t count = GetSelections(selections);
2840
2841 if ( count == 0 ||
2842 count > 1 ||
2843 HITEM(selections[0]) != htItem )
2844 {
2845 if ( HandleMouseEvent(nMsg, x, y, wParam) )
2846 {
2847 m_htClickedItem.Unset();
2848 break;
2849 }
2850
2851 // clear the previously selected items, if the user
2852 // clicked outside of the present selection, otherwise,
2853 // perform the deselection on mouse-up, this allows
2854 // multiple drag and drop to work.
2855 if ( !IsItemSelected(GetHwnd(), htItem))
2856 {
2857 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2858 this, htItem);
2859 changingEvent.m_itemOld = htOldItem;
2860
2861 if ( IsTreeEventAllowed(changingEvent) )
2862 {
2863 DoUnselectAll();
2864 DoSelectItem(wxTreeItemId(htItem));
2865 SetFocusedItem(wxTreeItemId(htItem));
2866
2867 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2868 this, htItem);
2869 changedEvent.m_itemOld = htOldItem;
2870 (void)HandleTreeEvent(changedEvent);
2871 }
2872 }
2873 else
2874 {
2875 SetFocusedItem(wxTreeItemId(htItem));
2876 m_mouseUpDeselect = true;
2877 }
2878 }
2879 else // click on a single selected item
2880 {
2881 // don't interfere with the default processing in
2882 // WM_MOUSEMOVE handler below as the default window
2883 // proc will start the drag itself if we let have
2884 // WM_LBUTTONDOWN
2885 m_htClickedItem.Unset();
2886
2887 // prevent in-place editing from starting if focus lost
2888 // since previous click
2889 if ( m_focusLost )
2890 {
2891 ClearFocusedItem();
2892 DoSelectItem(wxTreeItemId(htItem));
2893 SetFocusedItem(wxTreeItemId(htItem));
2894 }
2895 else
2896 {
2897 processed = false;
2898 }
2899 }
2900
2901 // reset on any click without Shift
2902 m_htSelStart.Unset();
2903 }
2904
2905 m_focusLost = false;
2906
2907 // we consumed the event so we need to trigger state image
2908 // click if needed
2909 if ( processed )
2910 {
2911 int htFlags = 0;
2912 wxTreeItemId item = HitTest(wxPoint(x, y), htFlags);
2913
2914 if ( htFlags & wxTREE_HITTEST_ONITEMSTATEICON )
2915 {
2916 m_triggerStateImageClick = true;
2917 }
2918 }
2919 break;
2920
2921 case WM_RBUTTONDOWN:
2922 if ( !isMultiple )
2923 break;
2924
2925 processed = true;
2926 SetFocus();
2927
2928 if ( HandleMouseEvent(nMsg, x, y, wParam) || !htItem )
2929 {
2930 break;
2931 }
2932
2933 // default handler removes the highlight from the currently
2934 // focused item when right mouse button is pressed on another
2935 // one but keeps the remaining items highlighted, which is
2936 // confusing, so override this default behaviour
2937 if ( !IsItemSelected(GetHwnd(), htItem) )
2938 {
2939 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
2940 this, htItem);
2941 changingEvent.m_itemOld = htOldItem;
2942
2943 if ( IsTreeEventAllowed(changingEvent) )
2944 {
2945 DoUnselectAll();
2946 DoSelectItem(wxTreeItemId(htItem));
2947 SetFocusedItem(wxTreeItemId(htItem));
2948
2949 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
2950 this, htItem);
2951 changedEvent.m_itemOld = htOldItem;
2952 (void)HandleTreeEvent(changedEvent);
2953 }
2954 }
2955
2956 break;
2957
2958 case WM_MOUSEMOVE:
2959#ifndef __WXWINCE__
2960 if ( m_htClickedItem )
2961 {
2962 int cx = abs(m_ptClick.x - x);
2963 int cy = abs(m_ptClick.y - y);
2964
2965 if ( cx > ::GetSystemMetrics(SM_CXDRAG) ||
2966 cy > ::GetSystemMetrics(SM_CYDRAG) )
2967 {
2968 NM_TREEVIEW tv;
2969 wxZeroMemory(tv);
2970
2971 tv.hdr.hwndFrom = GetHwnd();
2972 tv.hdr.idFrom = ::GetWindowLong(GetHwnd(), GWL_ID);
2973 tv.hdr.code = TVN_BEGINDRAG;
2974
2975 tv.itemNew.hItem = HITEM(m_htClickedItem);
2976
2977
2978 TVITEM tviAux;
2979 wxZeroMemory(tviAux);
2980
2981 tviAux.hItem = HITEM(m_htClickedItem);
2982 tviAux.mask = TVIF_STATE | TVIF_PARAM;
2983 tviAux.stateMask = 0xffffffff;
2984 TreeView_GetItem(GetHwnd(), &tviAux);
2985
2986 tv.itemNew.state = tviAux.state;
2987 tv.itemNew.lParam = tviAux.lParam;
2988
2989 tv.ptDrag.x = x;
2990 tv.ptDrag.y = y;
2991
2992 // do it before SendMessage() call below to avoid
2993 // reentrancies here if there is another WM_MOUSEMOVE
2994 // in the queue already
2995 m_htClickedItem.Unset();
2996
2997 ::SendMessage(GetHwndOf(GetParent()), WM_NOTIFY,
2998 tv.hdr.idFrom, (LPARAM)&tv );
2999
3000 // don't pass it to the default window proc, it would
3001 // start dragging again
3002 processed = true;
3003 }
3004 }
3005#endif // __WXWINCE__
3006
3007#if wxUSE_DRAGIMAGE
3008 if ( m_dragImage )
3009 {
3010 m_dragImage->Move(wxPoint(x, y));
3011 if ( htItem )
3012 {
3013 // highlight the item as target (hiding drag image is
3014 // necessary - otherwise the display will be corrupted)
3015 m_dragImage->Hide();
3016 TreeView_SelectDropTarget(GetHwnd(), htItem);
3017 m_dragImage->Show();
3018 }
3019 }
3020#endif // wxUSE_DRAGIMAGE
3021 break;
3022
3023 case WM_LBUTTONUP:
3024 if ( isMultiple )
3025 {
3026 // deselect other items if needed
3027 if ( htItem )
3028 {
3029 if ( m_mouseUpDeselect )
3030 {
3031 m_mouseUpDeselect = false;
3032
3033 wxTreeEvent changingEvent(wxEVT_COMMAND_TREE_SEL_CHANGING,
3034 this, htItem);
3035 changingEvent.m_itemOld = htOldItem;
3036
3037 if ( IsTreeEventAllowed(changingEvent) )
3038 {
3039 DoUnselectAll();
3040 DoSelectItem(wxTreeItemId(htItem));
3041 SetFocusedItem(wxTreeItemId(htItem));
3042
3043 wxTreeEvent changedEvent(wxEVT_COMMAND_TREE_SEL_CHANGED,
3044 this, htItem);
3045 changedEvent.m_itemOld = htOldItem;
3046 (void)HandleTreeEvent(changedEvent);
3047 }
3048 }
3049 }
3050
3051 m_htClickedItem.Unset();
3052
3053 if ( m_triggerStateImageClick )
3054 {
3055 if ( tvht.flags & TVHT_ONITEMSTATEICON )
3056 {
3057 wxTreeEvent event(wxEVT_COMMAND_TREE_STATE_IMAGE_CLICK,
3058 this, htItem);
3059 (void)HandleTreeEvent(event);
3060
3061 m_triggerStateImageClick = false;
3062 processed = true;
3063 }
3064 }
3065
3066 if ( !m_dragStarted && MSWIsOnItem(tvht.flags) )
3067 {
3068 processed = true;
3069 }
3070 }
3071
3072 // fall through
3073
3074 case WM_RBUTTONUP:
3075#if wxUSE_DRAGIMAGE
3076 if ( m_dragImage )
3077 {
3078 m_dragImage->EndDrag();
3079 wxDELETE(m_dragImage);
3080
3081 // generate the drag end event
3082 wxTreeEvent event(wxEVT_COMMAND_TREE_END_DRAG,
3083 this, htItem);
3084 event.m_pointDrag = wxPoint(x, y);
3085 (void)HandleTreeEvent(event);
3086
3087 // if we don't do it, the tree seems to think that 2 items
3088 // are selected simultaneously which is quite weird
3089 TreeView_SelectDropTarget(GetHwnd(), 0);
3090 }
3091#endif // wxUSE_DRAGIMAGE
3092
3093 if ( isMultiple && nMsg == WM_RBUTTONUP )
3094 {
3095 // send NM_RCLICK
3096 NMHDR nmhdr;
3097 nmhdr.hwndFrom = GetHwnd();
3098 nmhdr.idFrom = ::GetWindowLong(GetHwnd(), GWL_ID);
3099 nmhdr.code = NM_RCLICK;
3100 ::SendMessage(::GetParent(GetHwnd()), WM_NOTIFY,
3101 nmhdr.idFrom, (LPARAM)&nmhdr);
3102 processed = true;
3103 }
3104
3105 m_dragStarted = false;
3106
3107 break;
3108 }
3109 }
3110 else if ( (nMsg == WM_SETFOCUS || nMsg == WM_KILLFOCUS) )
3111 {
3112 if ( isMultiple )
3113 {
3114 // the tree control greys out the selected item when it loses focus
3115 // and paints it as selected again when it regains it, but it won't
3116 // do it for the other items itself - help it
3117 wxArrayTreeItemIds selections;
3118 size_t count = GetSelections(selections);
3119 TVGetItemRectParam param;
3120
3121 for ( size_t n = 0; n < count; n++ )
3122 {
3123 // TreeView_GetItemRect() will return false if item is not
3124 // visible, which may happen perfectly well
3125 if ( wxTreeView_GetItemRect(GetHwnd(), HITEM(selections[n]),
3126 param, TRUE) )
3127 {
3128 ::InvalidateRect(GetHwnd(), &param.rect, FALSE);
3129 }
3130 }
3131 }
3132
3133 if ( nMsg == WM_KILLFOCUS )
3134 {
3135 m_focusLost = true;
3136 }
3137 }
3138 else if ( (nMsg == WM_KEYDOWN || nMsg == WM_SYSKEYDOWN) && isMultiple )
3139 {
3140 // normally we want to generate wxEVT_KEY_DOWN events from TVN_KEYDOWN
3141 // notification but for the keys which can be used to change selection
3142 // we need to do it from here so as to not apply the default behaviour
3143 // if the events are handled by the user code
3144 switch ( wParam )
3145 {
3146 case VK_RETURN:
3147 case VK_SPACE:
3148 case VK_UP:
3149 case VK_DOWN:
3150 case VK_LEFT:
3151 case VK_RIGHT:
3152 case VK_HOME:
3153 case VK_END:
3154 case VK_PRIOR:
3155 case VK_NEXT:
3156 if ( !HandleKeyDown(wParam, lParam) &&
3157 !MSWHandleTreeKeyDownEvent(wParam, lParam) )
3158 {
3159 // use the key to update the selection if it was left
3160 // unprocessed
3161 MSWHandleSelectionKey(wParam);
3162 }
3163
3164 // pretend that we did process it in any case as we already
3165 // generated an event for it
3166 processed = true;
3167
3168 //default: for all the other keys leave processed as false so that
3169 // the tree control generates a TVN_KEYDOWN for us
3170 }
3171
3172 }
3173 else if ( nMsg == WM_COMMAND )
3174 {
3175 // if we receive a EN_KILLFOCUS command from the in-place edit control
3176 // used for label editing, make sure to end editing
3177 WORD id, cmd;
3178 WXHWND hwnd;
3179 UnpackCommand(wParam, lParam, &id, &hwnd, &cmd);
3180
3181 if ( cmd == EN_KILLFOCUS )
3182 {
3183 if ( m_textCtrl && m_textCtrl->GetHandle() == hwnd )
3184 {
3185 DoEndEditLabel();
3186
3187 processed = true;
3188 }
3189 }
3190 }
3191
3192 if ( !processed )
3193 rc = wxControl::MSWWindowProc(nMsg, wParam, lParam);
3194
3195 return rc;
3196}
3197
3198WXLRESULT
3199wxTreeCtrl::MSWDefWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
3200{
3201 if ( nMsg == WM_CHAR )
3202 {
3203 // don't let the control process Space and Return keys because it
3204 // doesn't do anything useful with them anyhow but always beeps
3205 // annoyingly when it receives them and there is no way to turn it off
3206 // simply if you just process TREEITEM_ACTIVATED event to which Space
3207 // and Enter presses are mapped in your code
3208 if ( wParam == VK_SPACE || wParam == VK_RETURN )
3209 return 0;
3210 }
3211#if wxUSE_DRAGIMAGE
3212 else if ( nMsg == WM_KEYDOWN )
3213 {
3214 if ( wParam == VK_ESCAPE )
3215 {
3216 if ( m_dragImage )
3217 {
3218 m_dragImage->EndDrag();
3219 wxDELETE(m_dragImage);
3220
3221 // if we don't do it, the tree seems to think that 2 items
3222 // are selected simultaneously which is quite weird
3223 TreeView_SelectDropTarget(GetHwnd(), 0);
3224 }
3225 }
3226 }
3227#endif // wxUSE_DRAGIMAGE
3228
3229 return wxControl::MSWDefWindowProc(nMsg, wParam, lParam);
3230}
3231
3232// process WM_NOTIFY Windows message
3233bool wxTreeCtrl::MSWOnNotify(int idCtrl, WXLPARAM lParam, WXLPARAM *result)
3234{
3235 wxTreeEvent event(wxEVT_NULL, this);
3236 wxEventType eventType = wxEVT_NULL;
3237 NMHDR *hdr = (NMHDR *)lParam;
3238
3239 switch ( hdr->code )
3240 {
3241 case TVN_BEGINDRAG:
3242 eventType = wxEVT_COMMAND_TREE_BEGIN_DRAG;
3243 // fall through
3244
3245 case TVN_BEGINRDRAG:
3246 {
3247 if ( eventType == wxEVT_NULL )
3248 eventType = wxEVT_COMMAND_TREE_BEGIN_RDRAG;
3249 //else: left drag, already set above
3250
3251 NM_TREEVIEW *tv = (NM_TREEVIEW *)lParam;
3252
3253 event.m_item = tv->itemNew.hItem;
3254 event.m_pointDrag = wxPoint(tv->ptDrag.x, tv->ptDrag.y);
3255
3256 // don't allow dragging by default: the user code must
3257 // explicitly say that it wants to allow it to avoid breaking
3258 // the old apps
3259 event.Veto();
3260 }
3261 break;
3262
3263 case TVN_BEGINLABELEDIT:
3264 {
3265 eventType = wxEVT_COMMAND_TREE_BEGIN_LABEL_EDIT;
3266 TV_DISPINFO *info = (TV_DISPINFO *)lParam;
3267
3268 // although the user event handler may still veto it, it is
3269 // important to set it now so that calls to SetItemText() from
3270 // the event handler would change the text controls contents
3271 m_idEdited =
3272 event.m_item = info->item.hItem;
3273 event.m_label = info->item.pszText;
3274 event.m_editCancelled = false;
3275 }
3276 break;
3277
3278 case TVN_DELETEITEM:
3279 {
3280 eventType = wxEVT_COMMAND_TREE_DELETE_ITEM;
3281 NM_TREEVIEW *tv = (NM_TREEVIEW *)lParam;
3282
3283 event.m_item = tv->itemOld.hItem;
3284
3285 if ( m_hasAnyAttr )
3286 {
3287 wxMapTreeAttr::iterator it = m_attrs.find(tv->itemOld.hItem);
3288 if ( it != m_attrs.end() )
3289 {
3290 delete it->second;
3291 m_attrs.erase(it);
3292 }
3293 }
3294 }
3295 break;
3296
3297 case TVN_ENDLABELEDIT:
3298 {
3299 eventType = wxEVT_COMMAND_TREE_END_LABEL_EDIT;
3300 TV_DISPINFO *info = (TV_DISPINFO *)lParam;
3301
3302 event.m_item = info->item.hItem;
3303 event.m_label = info->item.pszText;
3304 event.m_editCancelled = info->item.pszText == NULL;
3305 break;
3306 }
3307
3308#ifndef __WXWINCE__
3309 // These *must* not be removed or TVN_GETINFOTIP will
3310 // not be processed each time the mouse is moved
3311 // and the tooltip will only ever update once.
3312 case TTN_NEEDTEXTA:
3313 case TTN_NEEDTEXTW:
3314 {
3315 *result = 0;
3316
3317 break;
3318 }
3319
3320#ifdef TVN_GETINFOTIP
3321 case TVN_GETINFOTIP:
3322 {
3323 eventType = wxEVT_COMMAND_TREE_ITEM_GETTOOLTIP;
3324 NMTVGETINFOTIP *info = (NMTVGETINFOTIP*)lParam;
3325
3326 // Which item are we trying to get a tooltip for?
3327 event.m_item = info->hItem;
3328
3329 break;
3330 }
3331#endif // TVN_GETINFOTIP
3332#endif // !__WXWINCE__
3333
3334 case TVN_GETDISPINFO:
3335 eventType = wxEVT_COMMAND_TREE_GET_INFO;
3336 // fall through
3337
3338 case TVN_SETDISPINFO:
3339 {
3340 if ( eventType == wxEVT_NULL )
3341 eventType = wxEVT_COMMAND_TREE_SET_INFO;
3342 //else: get, already set above
3343
3344 TV_DISPINFO *info = (TV_DISPINFO *)lParam;
3345
3346 event.m_item = info->item.hItem;
3347 break;
3348 }
3349
3350 case TVN_ITEMEXPANDING:
3351 case TVN_ITEMEXPANDED:
3352 {
3353 NM_TREEVIEW *tv = (NM_TREEVIEW*)lParam;
3354
3355 int what;
3356 switch ( tv->action )
3357 {
3358 default:
3359 wxLogDebug(wxT("unexpected code %d in TVN_ITEMEXPAND message"), tv->action);
3360 // fall through
3361
3362 case TVE_EXPAND:
3363 what = IDX_EXPAND;
3364 break;
3365
3366 case TVE_COLLAPSE:
3367 what = IDX_COLLAPSE;
3368 break;
3369 }
3370
3371 int how = hdr->code == TVN_ITEMEXPANDING ? IDX_DOING
3372 : IDX_DONE;
3373
3374 eventType = gs_expandEvents[what][how];
3375
3376 event.m_item = tv->itemNew.hItem;
3377 }
3378 break;
3379
3380 case TVN_KEYDOWN:
3381 {
3382 TV_KEYDOWN *info = (TV_KEYDOWN *)lParam;
3383
3384 // fabricate the lParam and wParam parameters sufficiently
3385 // similar to the ones from a "real" WM_KEYDOWN so that
3386 // CreateKeyEvent() works correctly
3387 return MSWHandleTreeKeyDownEvent(
3388 info->wVKey, (wxIsAltDown() ? KF_ALTDOWN : 0) << 16);
3389 }
3390
3391
3392 // Vista's tree control has introduced some problems with our
3393 // multi-selection tree. When TreeView_SelectItem() is called,
3394 // the wrong items are deselected.
3395
3396 // Fortunately, Vista provides a new notification, TVN_ITEMCHANGING
3397 // that can be used to regulate this incorrect behaviour. The
3398 // following messages will allow only the unlocked item's selection
3399 // state to change
3400
3401 case TVN_ITEMCHANGINGA:
3402 case TVN_ITEMCHANGINGW:
3403 {
3404 // we only need to handles these in multi-select trees
3405 if ( HasFlag(wxTR_MULTIPLE) )
3406 {
3407 // get info about the item about to be changed
3408 NMTVITEMCHANGE* info = (NMTVITEMCHANGE*)lParam;
3409 if (TreeItemUnlocker::IsLocked(info->hItem))
3410 {
3411 // item's state is locked, don't allow the change
3412 // returning 1 will disallow the change
3413 *result = 1;
3414 return true;
3415 }
3416 }
3417
3418 // allow the state change
3419 }
3420 return false;
3421
3422 // NB: MSLU is broken and sends TVN_SELCHANGEDA instead of
3423 // TVN_SELCHANGEDW in Unicode mode under Win98. Therefore
3424 // we have to handle both messages:
3425 case TVN_SELCHANGEDA:
3426 case TVN_SELCHANGEDW:
3427 if ( !m_changingSelection )
3428 {
3429 eventType = wxEVT_COMMAND_TREE_SEL_CHANGED;
3430 }
3431 // fall through
3432
3433 case TVN_SELCHANGINGA:
3434 case TVN_SELCHANGINGW:
3435 if ( !m_changingSelection )
3436 {
3437 if ( eventType == wxEVT_NULL )
3438 eventType = wxEVT_COMMAND_TREE_SEL_CHANGING;
3439 //else: already set above
3440
3441 if (hdr->code == TVN_SELCHANGINGW ||
3442 hdr->code == TVN_SELCHANGEDW)
3443 {
3444 NM_TREEVIEWW *tv = (NM_TREEVIEWW *)lParam;
3445 event.m_item = tv->itemNew.hItem;
3446 event.m_itemOld = tv->itemOld.hItem;
3447 }
3448 else
3449 {
3450 NM_TREEVIEWA *tv = (NM_TREEVIEWA *)lParam;
3451 event.m_item = tv->itemNew.hItem;
3452 event.m_itemOld = tv->itemOld.hItem;
3453 }
3454 }
3455
3456 // we receive this message from WM_LBUTTONDOWN handler inside
3457 // comctl32.dll and so before the click is passed to
3458 // DefWindowProc() which sets the focus to the window which was
3459 // clicked and this can lead to unexpected event sequences: for
3460 // example, we may get a "selection change" event from the tree
3461 // before getting a "kill focus" event for the text control which
3462 // had the focus previously, thus breaking user code doing input
3463 // validation
3464 //
3465 // to avoid such surprises, we force the generation of focus events
3466 // now, before we generate the selection change ones
3467 if ( !m_changingSelection )
3468 SetFocus();
3469 break;
3470
3471 // instead of explicitly checking for _WIN32_IE, check if the
3472 // required symbols are available in the headers
3473#if defined(CDDS_PREPAINT)
3474 case NM_CUSTOMDRAW:
3475 {
3476 LPNMTVCUSTOMDRAW lptvcd = (LPNMTVCUSTOMDRAW)lParam;
3477 NMCUSTOMDRAW& nmcd = lptvcd->nmcd;
3478 switch ( nmcd.dwDrawStage )
3479 {
3480 case CDDS_PREPAINT:
3481 // if we've got any items with non standard attributes,
3482 // notify us before painting each item
3483 *result = m_hasAnyAttr ? CDRF_NOTIFYITEMDRAW
3484 : CDRF_DODEFAULT;
3485
3486 // windows in TreeCtrl use one-based index for item state images,
3487 // 0 indexed image is not being used, we're using zero-based index,
3488 // so we have to add temp image (of zero index) to state image list
3489 // before we draw any item, then after items are drawn we have to
3490 // delete it (in POSTPAINT notify)
3491 if (m_imageListState && m_imageListState->GetImageCount() > 0)
3492 {
3493 typedef BOOL (wxSTDCALL *ImageList_Copy_t)
3494 (HIMAGELIST, int, HIMAGELIST, int, UINT);
3495 static ImageList_Copy_t s_pfnImageList_Copy = NULL;
3496 static bool loaded = false;
3497
3498 if ( !loaded )
3499 {
3500 wxLoadedDLL dllComCtl32(wxT("comctl32.dll"));
3501 if ( dllComCtl32.IsLoaded() )
3502 wxDL_INIT_FUNC(s_pfn, ImageList_Copy, dllComCtl32);
3503 }
3504
3505 if ( !s_pfnImageList_Copy )
3506 {
3507 // this code is broken with ImageList_Copy()
3508 // but I don't care enough about Win95 support
3509 // to write it now -- if anybody does, please
3510 // do it
3511 wxFAIL_MSG("TODO: implement this for Win95");
3512 break;
3513 }
3514
3515 const HIMAGELIST
3516 hImageList = GetHimagelistOf(m_imageListState);
3517
3518 // add temporary image
3519 int width, height;
3520 m_imageListState->GetSize(0, width, height);
3521
3522 HBITMAP hbmpTemp = ::CreateBitmap(width, height, 1, 1, NULL);
3523 int index = ::ImageList_Add(hImageList, hbmpTemp, hbmpTemp);
3524 ::DeleteObject(hbmpTemp);
3525
3526 if ( index != -1 )
3527 {
3528 // move images to right
3529 for ( int i = index; i > 0; i-- )
3530 {
3531 (*s_pfnImageList_Copy)(hImageList, i,
3532 hImageList, i-1,
3533 ILCF_MOVE);
3534 }
3535
3536 // we must remove the image in POSTPAINT notify
3537 *result |= CDRF_NOTIFYPOSTPAINT;
3538 }
3539 }
3540 break;
3541
3542 case CDDS_POSTPAINT:
3543 // we are deleting temp image of 0 index, which was
3544 // added before items were drawn (in PREPAINT notify)
3545 if (m_imageListState && m_imageListState->GetImageCount() > 0)
3546 m_imageListState->Remove(0);
3547 break;
3548
3549 case CDDS_ITEMPREPAINT:
3550 {
3551 wxMapTreeAttr::iterator
3552 it = m_attrs.find((void *)nmcd.dwItemSpec);
3553
3554 if ( it == m_attrs.end() )
3555 {
3556 // nothing to do for this item
3557 *result = CDRF_DODEFAULT;
3558 break;
3559 }
3560
3561 wxTreeItemAttr * const attr = it->second;
3562
3563 wxTreeViewItem tvItem((void *)nmcd.dwItemSpec,
3564 TVIF_STATE, TVIS_DROPHILITED);
3565 DoGetItem(&tvItem);
3566 const UINT tvItemState = tvItem.state;
3567
3568 // selection colours should override ours,
3569 // otherwise it is too confusing to the user
3570 if ( !(nmcd.uItemState & CDIS_SELECTED) &&
3571 !(tvItemState & TVIS_DROPHILITED) )
3572 {
3573 wxColour colBack;
3574 if ( attr->HasBackgroundColour() )
3575 {
3576 colBack = attr->GetBackgroundColour();
3577 lptvcd->clrTextBk = wxColourToRGB(colBack);
3578 }
3579 }
3580
3581 // but we still want to keep the special foreground
3582 // colour when we don't have focus (we can't keep
3583 // it when we do, it would usually be unreadable on
3584 // the almost inverted bg colour...)
3585 if ( ( !(nmcd.uItemState & CDIS_SELECTED) ||
3586 FindFocus() != this ) &&
3587 !(tvItemState & TVIS_DROPHILITED) )
3588 {
3589 wxColour colText;
3590 if ( attr->HasTextColour() )
3591 {
3592 colText = attr->GetTextColour();
3593 lptvcd->clrText = wxColourToRGB(colText);
3594 }
3595 }
3596
3597 if ( attr->HasFont() )
3598 {
3599 HFONT hFont = GetHfontOf(attr->GetFont());
3600
3601 ::SelectObject(nmcd.hdc, hFont);
3602
3603 *result = CDRF_NEWFONT;
3604 }
3605 else // no specific font
3606 {
3607 *result = CDRF_DODEFAULT;
3608 }
3609 }
3610 break;
3611
3612 default:
3613 *result = CDRF_DODEFAULT;
3614 }
3615 }
3616
3617 // we always process it
3618 return true;
3619#endif // have owner drawn support in headers
3620
3621 case NM_CLICK:
3622 {
3623 DWORD pos = GetMessagePos();
3624 POINT point;
3625 point.x = LOWORD(pos);
3626 point.y = HIWORD(pos);
3627 ::MapWindowPoints(HWND_DESKTOP, GetHwnd(), &point, 1);
3628 int htFlags = 0;
3629 wxTreeItemId item = HitTest(wxPoint(point.x, point.y), htFlags);
3630
3631 if ( htFlags & wxTREE_HITTEST_ONITEMSTATEICON )
3632 {
3633 event.m_item = item;
3634 eventType = wxEVT_COMMAND_TREE_STATE_IMAGE_CLICK;
3635 }
3636
3637 break;
3638 }
3639
3640 case NM_DBLCLK:
3641 case NM_RCLICK:
3642 {
3643 TV_HITTESTINFO tvhti;
3644 ::GetCursorPos(&tvhti.pt);
3645 ::ScreenToClient(GetHwnd(), &tvhti.pt);
3646 if ( TreeView_HitTest(GetHwnd(), &tvhti) )
3647 {
3648 if ( MSWIsOnItem(tvhti.flags) )
3649 {
3650 event.m_item = tvhti.hItem;
3651 eventType = (int)hdr->code == NM_DBLCLK
3652 ? wxEVT_COMMAND_TREE_ITEM_ACTIVATED
3653 : wxEVT_COMMAND_TREE_ITEM_RIGHT_CLICK;
3654
3655 event.m_pointDrag.x = tvhti.pt.x;
3656 event.m_pointDrag.y = tvhti.pt.y;
3657 }
3658
3659 break;
3660 }
3661 }
3662 // fall through
3663
3664 default:
3665 return wxControl::MSWOnNotify(idCtrl, lParam, result);
3666 }
3667
3668 event.SetEventType(eventType);
3669
3670 bool processed = HandleTreeEvent(event);
3671
3672 // post processing
3673 switch ( hdr->code )
3674 {
3675 case NM_DBLCLK:
3676 // we translate NM_DBLCLK into ACTIVATED event and if the user
3677 // handled the activation of the item we shouldn't proceed with
3678 // also using the same double click for toggling the item expanded
3679 // state -- but OTOH do let the user to expand/collapse the item by
3680 // double clicking on it if the activation is not handled specially
3681 *result = processed;
3682 break;
3683
3684 case NM_RCLICK:
3685 // prevent tree control from sending WM_CONTEXTMENU to our parent
3686 // (which it does if NM_RCLICK is not handled) because we want to
3687 // send it to the control itself
3688 *result =
3689 processed = true;
3690
3691 ::SendMessage(GetHwnd(), WM_CONTEXTMENU,
3692 (WPARAM)GetHwnd(), ::GetMessagePos());
3693 break;
3694
3695 case TVN_BEGINDRAG:
3696 case TVN_BEGINRDRAG:
3697#if wxUSE_DRAGIMAGE
3698 if ( event.IsAllowed() )
3699 {
3700 // normally this is impossible because the m_dragImage is
3701 // deleted once the drag operation is over
3702 wxASSERT_MSG( !m_dragImage, wxT("starting to drag once again?") );
3703
3704 m_dragImage = new wxDragImage(*this, event.m_item);
3705 m_dragImage->BeginDrag(wxPoint(0,0), this);
3706 m_dragImage->Show();
3707
3708 m_dragStarted = true;
3709 }
3710#endif // wxUSE_DRAGIMAGE
3711 break;
3712
3713 case TVN_DELETEITEM:
3714 {
3715 // NB: we might process this message using wxWidgets event
3716 // tables, but due to overhead of wxWin event system we
3717 // prefer to do it here ourself (otherwise deleting a tree
3718 // with many items is just too slow)
3719 NM_TREEVIEW *tv = (NM_TREEVIEW *)lParam;
3720
3721 wxTreeItemParam *param =
3722 (wxTreeItemParam *)tv->itemOld.lParam;
3723 delete param;
3724
3725 processed = true; // Make sure we don't get called twice
3726 }
3727 break;
3728
3729 case TVN_BEGINLABELEDIT:
3730 // return true to cancel label editing
3731 *result = !event.IsAllowed();
3732
3733 // set ES_WANTRETURN ( like we do in BeginLabelEdit )
3734 if ( event.IsAllowed() )
3735 {
3736 HWND hText = TreeView_GetEditControl(GetHwnd());
3737 if ( hText )
3738 {
3739 // MBN: if m_textCtrl already has an HWND, it is a stale
3740 // pointer from a previous edit (because the user
3741 // didn't modify the label before dismissing the control,
3742 // and TVN_ENDLABELEDIT was not sent), so delete it
3743 if ( m_textCtrl && m_textCtrl->GetHWND() )
3744 DeleteTextCtrl();
3745 if ( !m_textCtrl )
3746 m_textCtrl = new wxTextCtrl();
3747 m_textCtrl->SetParent(this);
3748 m_textCtrl->SetHWND((WXHWND)hText);
3749 m_textCtrl->SubclassWin((WXHWND)hText);
3750
3751 // set wxTE_PROCESS_ENTER style for the text control to
3752 // force it to process the Enter presses itself, otherwise
3753 // they could be stolen from it by the dialog
3754 // navigation code
3755 m_textCtrl->SetWindowStyle(m_textCtrl->GetWindowStyle()
3756 | wxTE_PROCESS_ENTER);
3757 }
3758 }
3759 else // we had set m_idEdited before
3760 {
3761 m_idEdited.Unset();
3762 }
3763 break;
3764
3765 case TVN_ENDLABELEDIT:
3766 // return true to set the label to the new string: note that we
3767 // also must pretend that we did process the message or it is going
3768 // to be passed to DefWindowProc() which will happily return false
3769 // cancelling the label change
3770 *result = event.IsAllowed();
3771 processed = true;
3772
3773 // ensure that we don't have the text ctrl which is going to be
3774 // deleted any more
3775 DeleteTextCtrl();
3776 break;
3777
3778#ifndef __WXWINCE__
3779#ifdef TVN_GETINFOTIP
3780 case TVN_GETINFOTIP:
3781 {
3782 // If the user permitted a tooltip change, change it
3783 if (event.IsAllowed())
3784 {
3785 SetToolTip(event.m_label);
3786 }
3787 }
3788 break;
3789#endif
3790#endif
3791
3792 case TVN_SELCHANGING:
3793 case TVN_ITEMEXPANDING:
3794 // return true to prevent the action from happening
3795 *result = !event.IsAllowed();
3796 break;
3797
3798 case TVN_ITEMEXPANDED:
3799 {
3800 NM_TREEVIEW *tv = (NM_TREEVIEW *)lParam;
3801 const wxTreeItemId id(tv->itemNew.hItem);
3802
3803 if ( tv->action == TVE_COLLAPSE )
3804 {
3805 if ( wxApp::GetComCtl32Version() >= 600 )
3806 {
3807 // for some reason the item selection rectangle depends
3808 // on whether it is expanded or collapsed (at least
3809 // with comctl32.dll v6): it is wider (by 3 pixels) in
3810 // the expanded state, so when the item collapses and
3811 // then is deselected the rightmost 3 pixels of the
3812 // previously drawn selection are left on the screen
3813 //
3814 // it's not clear if it's a bug in comctl32.dll or in
3815 // our code (because it does not happen in Explorer but
3816 // OTOH we don't do anything which could result in this
3817 // AFAICS) but we do need to work around it to avoid
3818 // ugly artifacts
3819 RefreshItem(id);
3820 }
3821 }
3822 else // expand
3823 {
3824 // the item is also not refreshed properly after expansion when
3825 // it has an image depending on the expanded/collapsed state:
3826 // again, it's not clear if the bug is in comctl32.dll or our
3827 // code...
3828 int image = GetItemImage(id, wxTreeItemIcon_Expanded);
3829 if ( image != -1 )
3830 {
3831 RefreshItem(id);
3832 }
3833 }
3834 }
3835 break;
3836
3837 case TVN_GETDISPINFO:
3838 // NB: so far the user can't set the image himself anyhow, so do it
3839 // anyway - but this may change later
3840 //if ( /* !processed && */ )
3841 {
3842 wxTreeItemId item = event.m_item;
3843 TV_DISPINFO *info = (TV_DISPINFO *)lParam;
3844
3845 const wxTreeItemParam * const param = GetItemParam(item);
3846 if ( !param )
3847 break;
3848
3849 if ( info->item.mask & TVIF_IMAGE )
3850 {
3851 info->item.iImage =
3852 param->GetImage
3853 (
3854 IsExpanded(item) ? wxTreeItemIcon_Expanded
3855 : wxTreeItemIcon_Normal
3856 );
3857 }
3858 if ( info->item.mask & TVIF_SELECTEDIMAGE )
3859 {
3860 info->item.iSelectedImage =
3861 param->GetImage
3862 (
3863 IsExpanded(item) ? wxTreeItemIcon_SelectedExpanded
3864 : wxTreeItemIcon_Selected
3865 );
3866 }
3867 }
3868 break;
3869
3870 //default:
3871 // for the other messages the return value is ignored and there is
3872 // nothing special to do
3873 }
3874 return processed;
3875}
3876
3877// ----------------------------------------------------------------------------
3878// State control.
3879// ----------------------------------------------------------------------------
3880
3881// why do they define INDEXTOSTATEIMAGEMASK but not the inverse?
3882#define STATEIMAGEMASKTOINDEX(state) (((state) & TVIS_STATEIMAGEMASK) >> 12)
3883
3884int wxTreeCtrl::DoGetItemState(const wxTreeItemId& item) const
3885{
3886 wxCHECK_MSG( item.IsOk(), wxTREE_ITEMSTATE_NONE, wxT("invalid tree item") );
3887
3888 // receive the desired information
3889 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_STATEIMAGEMASK);
3890 DoGetItem(&tvItem);
3891
3892 // state images are one-based
3893 return STATEIMAGEMASKTOINDEX(tvItem.state) - 1;
3894}
3895
3896void wxTreeCtrl::DoSetItemState(const wxTreeItemId& item, int state)
3897{
3898 wxCHECK_RET( item.IsOk(), wxT("invalid tree item") );
3899
3900 wxTreeViewItem tvItem(item, TVIF_STATE, TVIS_STATEIMAGEMASK);
3901
3902 // state images are one-based
3903 // 0 if no state image display (wxTREE_ITEMSTATE_NONE = -1)
3904 tvItem.state = INDEXTOSTATEIMAGEMASK(state + 1);
3905
3906 DoSetItem(&tvItem);
3907}
3908
3909#endif // wxUSE_TREECTRL