]> git.saurik.com Git - wxWidgets.git/blame - src/generic/treectrl.cpp
Added #include to get FIONBIO on Solaris 2.6
[wxWidgets.git] / src / generic / treectrl.cpp
CommitLineData
c801d85f
KB
1/////////////////////////////////////////////////////////////////////////////
2// Name: treectrl.cpp
3// Purpose:
4// Author: Robert Roebling
5// Created: 01/02/97
389cdc7a 6// Id: $Id$
c801d85f 7// Copyright: (c) 1998 Robert Roebling, Julian Smart and Markus Holzem
29d87bba 8// Licence: wxWindows licence
c801d85f
KB
9/////////////////////////////////////////////////////////////////////////////
10
11#ifdef __GNUG__
12#pragma implementation "treectrl.h"
13#endif
14
15#include "wx/treectrl.h"
16#include "wx/settings.h"
389cdc7a 17#include "wx/log.h"
f4e325b3 18#include <wx/intl.h>
c801d85f
KB
19
20//-----------------------------------------------------------------------------
21// wxTreeItem
22//-----------------------------------------------------------------------------
23
24IMPLEMENT_DYNAMIC_CLASS(wxTreeItem, wxObject)
25
74bedbeb 26wxTreeItem::wxTreeItem()
c801d85f
KB
27{
28 m_mask = 0;
29 m_itemId = 0;
30 m_state = 0;
31 m_stateMask = 0;
32 m_image = -1;
33 m_selectedImage = -1;
34 m_children = 0;
35 m_data = 0;
edaa81ae 36}
c801d85f
KB
37
38//-----------------------------------------------------------------------------
39// wxTreeEvent
40//-----------------------------------------------------------------------------
41
42IMPLEMENT_DYNAMIC_CLASS(wxTreeEvent,wxCommandEvent)
43
7798a18e 44wxTreeEvent::wxTreeEvent( wxEventType commandType, int id ) :
c801d85f
KB
45 wxCommandEvent( commandType, id )
46{
47 m_code = 0;
48 m_oldItem = 0;
edaa81ae 49}
c801d85f
KB
50
51//-----------------------------------------------------------------------------
52// wxGenericTreeItem
53//-----------------------------------------------------------------------------
54
55IMPLEMENT_DYNAMIC_CLASS(wxGenericTreeItem,wxObject)
56
57wxGenericTreeItem::wxGenericTreeItem( wxGenericTreeItem *parent )
58{
59 Reset();
60 m_parent = parent;
61 m_hasHilight = FALSE;
edaa81ae 62}
c801d85f
KB
63
64wxGenericTreeItem::wxGenericTreeItem( wxGenericTreeItem *parent, const wxTreeItem& item, wxDC *dc )
65{
66 Reset();
67 SetItem( item, dc );
68 m_parent = parent;
69 m_hasHilight = FALSE;
edaa81ae 70}
c801d85f
KB
71
72void wxGenericTreeItem::SetItem( const wxTreeItem &item, wxDC *dc )
73{
29d87bba 74 if ((item.m_mask & wxTREE_MASK_HANDLE) == wxTREE_MASK_HANDLE)
c801d85f
KB
75 m_itemId = item.m_itemId;
76 if ((item.m_mask & wxTREE_MASK_STATE) == wxTREE_MASK_STATE)
77 m_state = item.m_state;
78 if ((item.m_mask & wxTREE_MASK_TEXT) == wxTREE_MASK_TEXT)
79 m_text = item.m_text;
80 if ((item.m_mask & wxTREE_MASK_IMAGE) == wxTREE_MASK_IMAGE)
81 m_image = item.m_image;
82 if ((item.m_mask & wxTREE_MASK_SELECTED_IMAGE) == wxTREE_MASK_SELECTED_IMAGE)
83 m_selectedImage = item.m_selectedImage;
84 if ((item.m_mask & wxTREE_MASK_CHILDREN) == wxTREE_MASK_CHILDREN)
85 m_hasChildren = (item.m_children > 0);
86 if ((item.m_mask & wxTREE_MASK_DATA) == wxTREE_MASK_DATA)
87 m_data = item.m_data;
88 long lw = 0;
89 long lh = 0;
90 dc->GetTextExtent( m_text, &lw, &lh );
91 m_width = lw;
92 m_height = lh;
edaa81ae 93}
c801d85f
KB
94
95void wxGenericTreeItem::SetText( const wxString &text, wxDC *dc )
96{
97 m_text = text;
98 long lw = 0;
99 long lh = 0;
100 dc->GetTextExtent( m_text, &lw, &lh );
101 m_width = lw;
102 m_height = lh;
edaa81ae 103}
c801d85f 104
74bedbeb 105void wxGenericTreeItem::Reset()
c801d85f
KB
106{
107 m_itemId = -1;
108 m_state = 0;
109 m_text = "";
110 m_image = -1;
111 m_selectedImage = -1;
112// m_children = 0;
113 m_hasChildren = FALSE;
114 m_data = 0;
115 m_x = 0;
116 m_y = 0;
117 m_height = 0;
118 m_width = 0;
119 m_xCross = 0;
120 m_yCross = 0;
121 m_level = 0;
122 m_children.DeleteContents( TRUE );
74bedbeb 123 m_isCollapsed = TRUE;
c67daf87 124 m_parent = (wxGenericTreeItem *) NULL;
edaa81ae 125}
c801d85f
KB
126
127void wxGenericTreeItem::GetItem( wxTreeItem &item ) const
128{
129 if ((item.m_mask & wxTREE_MASK_STATE) == wxTREE_MASK_STATE)
130 item.m_state = m_state;
131 if ((item.m_mask & wxTREE_MASK_TEXT) == wxTREE_MASK_TEXT)
132 item.m_text = m_text;
133 if ((item.m_mask & wxTREE_MASK_IMAGE) == wxTREE_MASK_IMAGE)
134 item.m_image = m_image;
135 if ((item.m_mask & wxTREE_MASK_SELECTED_IMAGE) == wxTREE_MASK_SELECTED_IMAGE)
136 item.m_selectedImage = m_selectedImage;
137 if ((item.m_mask & wxTREE_MASK_CHILDREN) == wxTREE_MASK_CHILDREN)
138 item.m_children = (int)m_hasChildren;
139 if ((item.m_mask & wxTREE_MASK_DATA) == wxTREE_MASK_DATA)
140 item.m_data = m_data;
edaa81ae 141}
c801d85f 142
74bedbeb 143bool wxGenericTreeItem::HasChildren()
c801d85f
KB
144{
145 return m_hasChildren;
edaa81ae 146}
c801d85f 147
74bedbeb 148bool wxGenericTreeItem::HasPlus()
c801d85f 149{
74bedbeb
VZ
150 if ( !HasChildren() )
151 return FALSE;
152
153 return !IsExpanded();
edaa81ae 154}
c801d85f 155
74bedbeb 156int wxGenericTreeItem::NumberOfVisibleDescendents()
c801d85f
KB
157{
158 int ret = m_children.Number();
159 wxNode *node = m_children.First();
160 while (node)
161 {
162 wxGenericTreeItem *item = (wxGenericTreeItem*)node->Data();
163 ret += item->NumberOfVisibleDescendents();
164 node = node->Next();
edaa81ae 165 }
c801d85f 166 return ret;
edaa81ae 167}
c801d85f 168
74bedbeb 169int wxGenericTreeItem::NumberOfVisibleChildren()
c801d85f 170{
74bedbeb 171 return m_isCollapsed ? 0 : m_children.Number();
edaa81ae 172}
c801d85f
KB
173
174wxGenericTreeItem *wxGenericTreeItem::FindItem( long itemId ) const
175{
176 if (m_itemId == itemId) return (wxGenericTreeItem*)(this);
177 wxNode *node = m_children.First();
178 while (node)
179 {
180 wxGenericTreeItem *item = (wxGenericTreeItem*)node->Data();
181 wxGenericTreeItem *res = item->FindItem( itemId );
182 if (res) return (wxGenericTreeItem*)(res);
183 node = node->Next();
edaa81ae 184 }
c67daf87 185 return (wxGenericTreeItem *) NULL;
edaa81ae 186}
c801d85f
KB
187
188void wxGenericTreeItem::AddChild( wxGenericTreeItem *child )
189{
190 m_children.Append( child );
edaa81ae 191}
c801d85f
KB
192
193void wxGenericTreeItem::SetCross( int x, int y )
194{
195 m_xCross = x;
196 m_yCross = y;
edaa81ae 197}
c801d85f
KB
198
199void wxGenericTreeItem::GetSize( int &x, int &y )
200{
201 if (y < m_y + 10) y = m_y +10;
202 int width = m_x + m_width;
203 if (width > x) x = width;
204 wxNode *node = m_children.First();
205 while (node)
206 {
207 wxGenericTreeItem *item = (wxGenericTreeItem*)node->Data();
208 item->GetSize( x, y );
209 node = node->Next();
edaa81ae
RR
210 }
211}
c801d85f
KB
212
213long wxGenericTreeItem::HitTest( const wxPoint& point, int &flags )
214{
4c681997 215 if ((point.y > m_y) && (point.y < m_y+m_height))
c801d85f
KB
216 {
217 if ((point.x > m_xCross-5) &&
218 (point.x < m_xCross+5) &&
219 (point.y > m_yCross-5) &&
220 (point.y < m_yCross+5) &&
29d87bba 221 (m_hasChildren))
c801d85f
KB
222 {
223 flags = wxTREE_HITTEST_ONITEMBUTTON;
224 return m_itemId;
edaa81ae 225 }
c801d85f
KB
226 if ((point.x > m_x) && (point.x < m_x+m_width))
227 {
228 flags = wxTREE_HITTEST_ONITEMLABEL;
229 return m_itemId;
edaa81ae 230 }
c801d85f
KB
231 if (point.x > m_x)
232 {
233 flags = wxTREE_HITTEST_ONITEMRIGHT;
234 return m_itemId;
edaa81ae 235 }
c801d85f
KB
236 flags = wxTREE_HITTEST_ONITEMINDENT;
237 return m_itemId;
238 }
239 else
240 {
e2414cbe 241 if (!m_isCollapsed)
c801d85f 242 {
e2414cbe
RR
243 wxNode *node = m_children.First();
244 while (node)
245 {
246 wxGenericTreeItem *child = (wxGenericTreeItem*)node->Data();
247 long res = child->HitTest( point, flags );
248 if (res != -1) return res;
249 node = node->Next();
edaa81ae
RR
250 }
251 }
252 }
c801d85f 253 return -1;
edaa81ae 254}
c801d85f
KB
255
256void wxGenericTreeItem::PrepareEvent( wxTreeEvent &event )
257{
258 event.m_item.m_itemId = m_itemId;
259 event.m_item.m_state = m_state;
260 event.m_item.m_text = m_text;
261 event.m_item.m_image = m_image;
262 event.m_item.m_selectedImage = m_selectedImage;
263 event.m_item.m_children = (int)m_hasChildren;
264 event.m_item.m_data = m_data;
265 event.m_oldItem = 0;
266 event.m_code = 0;
267 event.m_pointDrag.x = 0;
268 event.m_pointDrag.y = 0;
edaa81ae 269}
c801d85f
KB
270
271void wxGenericTreeItem::SendKeyDown( wxWindow *target )
272{
273 wxTreeEvent event( wxEVT_COMMAND_TREE_KEY_DOWN, target->GetId() );
274 PrepareEvent( event );
275 event.SetEventObject( target );
276 target->ProcessEvent( event );
edaa81ae 277}
c801d85f
KB
278
279void wxGenericTreeItem::SendSelected( wxWindow *target )
280{
281 wxTreeEvent event( wxEVT_COMMAND_TREE_SEL_CHANGED, target->GetId() );
282 PrepareEvent( event );
283 event.SetEventObject( target );
284 target->ProcessEvent( event );
edaa81ae 285}
c801d85f
KB
286
287void wxGenericTreeItem::SendDelete( wxWindow *target )
288{
74bedbeb 289 wxTreeEvent event( wxEVT_COMMAND_TREE_DELETE_ITEM, target->GetId() );
c801d85f
KB
290 PrepareEvent( event );
291 event.SetEventObject( target );
292 target->ProcessEvent( event );
edaa81ae 293}
c801d85f
KB
294
295void wxGenericTreeItem::SendExpand( wxWindow *target )
296{
74bedbeb
VZ
297 m_isCollapsed = FALSE;
298
299 wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_EXPANDING, target->GetId() );
74bedbeb
VZ
300 event.SetEventObject( target );
301 PrepareEvent( event );
302 target->ProcessEvent( event );
303
304 event.SetEventType(wxEVT_COMMAND_TREE_ITEM_EXPANDED);
c801d85f 305 PrepareEvent( event );
74bedbeb 306 target->ProcessEvent( event );
edaa81ae 307}
74bedbeb
VZ
308
309void wxGenericTreeItem::SendCollapse( wxWindow *target )
310{
311 m_isCollapsed = TRUE;
312
4c681997 313 wxTreeEvent event( wxEVT_COMMAND_TREE_ITEM_COLLAPSING, target->GetId() );
c801d85f 314 event.SetEventObject( target );
74bedbeb
VZ
315 PrepareEvent( event );
316 target->ProcessEvent( event );
317
4c681997 318 event.SetEventType(wxEVT_COMMAND_TREE_ITEM_COLLAPSED);
74bedbeb 319 PrepareEvent( event );
c801d85f 320 target->ProcessEvent( event );
edaa81ae 321}
c801d85f
KB
322
323void wxGenericTreeItem::SetHilight( bool set )
324{
325 m_hasHilight = set;
edaa81ae 326}
c801d85f 327
74bedbeb 328bool wxGenericTreeItem::HasHilight()
c801d85f
KB
329{
330 return m_hasHilight;
edaa81ae 331}
c801d85f
KB
332
333//-----------------------------------------------------------------------------
334// wxTreeCtrl
335//-----------------------------------------------------------------------------
336
4c681997 337IMPLEMENT_DYNAMIC_CLASS(wxTreeCtrl,wxScrolledWindow)
c801d85f
KB
338
339BEGIN_EVENT_TABLE(wxTreeCtrl,wxScrolledWindow)
340 EVT_PAINT (wxTreeCtrl::OnPaint)
341 EVT_MOUSE_EVENTS (wxTreeCtrl::OnMouse)
342 EVT_CHAR (wxTreeCtrl::OnChar)
343 EVT_SET_FOCUS (wxTreeCtrl::OnSetFocus)
344 EVT_KILL_FOCUS (wxTreeCtrl::OnKillFocus)
345END_EVENT_TABLE()
346
74bedbeb 347wxTreeCtrl::wxTreeCtrl()
c801d85f 348{
c67daf87
UR
349 m_current = (wxGenericTreeItem *) NULL;
350 m_anchor = (wxGenericTreeItem *) NULL;
c801d85f
KB
351 m_hasFocus = FALSE;
352 m_xScroll = 0;
353 m_yScroll = 0;
354 m_lastId = 0;
355 m_lineHeight = 10;
356 m_indent = 15;
357 m_isCreated = FALSE;
c801d85f 358 m_hilightBrush = new wxBrush( wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT), wxSOLID );
c67daf87
UR
359 m_imageList = (wxImageList *) NULL;
360 m_smallImageList = (wxImageList *) NULL;
edaa81ae 361}
c801d85f 362
debe6624 363wxTreeCtrl::wxTreeCtrl(wxWindow *parent, wxWindowID id,
4c681997 364 const wxPoint& pos, const wxSize& size,
74bedbeb 365 long style, const wxString& name )
c801d85f 366{
c67daf87
UR
367 m_current = (wxGenericTreeItem *) NULL;
368 m_anchor = (wxGenericTreeItem *) NULL;
c801d85f
KB
369 m_hasFocus = FALSE;
370 m_xScroll = 0;
371 m_yScroll = 0;
372 m_lastId = 0;
373 m_lineHeight = 10;
374 m_indent = 15;
375 m_isCreated = FALSE;
c801d85f 376 m_hilightBrush = new wxBrush( wxSystemSettings::GetSystemColour(wxSYS_COLOUR_HIGHLIGHT), wxSOLID );
c67daf87
UR
377 m_imageList = (wxImageList *) NULL;
378 m_smallImageList = (wxImageList *) NULL;
c801d85f 379 Create( parent, id, pos, size, style, name );
edaa81ae 380}
c801d85f 381
74bedbeb 382wxTreeCtrl::~wxTreeCtrl()
c801d85f 383{
edaa81ae 384}
c801d85f 385
debe6624 386bool wxTreeCtrl::Create(wxWindow *parent, wxWindowID id,
4c681997
RR
387 const wxPoint& pos, const wxSize& size,
388 long style, const wxString& name )
c801d85f
KB
389{
390 wxScrolledWindow::Create( parent, id, pos, size, style, name );
391 SetBackgroundColour( *wxWHITE );
392 m_dottedPen = wxPen( *wxBLACK, 0, 0 );
393 return TRUE;
edaa81ae 394}
c801d85f 395
74bedbeb 396int wxTreeCtrl::GetCount() const
c801d85f
KB
397{
398 if (!m_anchor) return 0;
399 return m_anchor->NumberOfVisibleDescendents();
edaa81ae 400}
c801d85f 401
74bedbeb
VZ
402long wxTreeCtrl::InsertItem( long parent, const wxString& label, int image,
403 int selImage, long WXUNUSED(insertAfter) )
c801d85f 404{
c67daf87 405 wxGenericTreeItem *p = (wxGenericTreeItem *) NULL;
c801d85f
KB
406 if (parent == 0)
407 {
408 if (m_anchor) return -1;
409 }
410 else
411 {
412 p = FindItem( parent );
413 if (!p) return -1;
edaa81ae 414 }
c801d85f
KB
415 wxTreeItem item;
416 m_lastId++;
417 item.m_mask = wxTREE_MASK_HANDLE;
418 item.m_itemId = m_lastId;
419 if (!label.IsNull() || (label == ""))
420 {
421 item.m_text = label;
422 item.m_mask |= wxTREE_MASK_TEXT;
edaa81ae 423 }
c801d85f
KB
424 if (image >= 0)
425 {
426 item.m_image = image;
427 item.m_mask |= wxTREE_MASK_IMAGE;
edaa81ae 428 }
c801d85f
KB
429 if (selImage >= 0)
430 {
431 item.m_selectedImage = selImage;
432 item.m_mask |= wxTREE_MASK_SELECTED_IMAGE;
edaa81ae 433 }
29d87bba 434
c801d85f
KB
435 wxClientDC dc(this);
436 wxGenericTreeItem *new_child = new wxGenericTreeItem( p, item, &dc );
29d87bba 437 if (p)
c801d85f
KB
438 p->AddChild( new_child );
439 else
440 m_anchor = new_child;
29d87bba 441
c801d85f
KB
442 if (p)
443 {
444 CalculatePositions();
29d87bba 445
e2414cbe 446 if (!p->HasChildren()) p->m_hasChildren = TRUE;
389cdc7a 447
c801d85f 448 int ch = 0;
c67daf87 449 GetClientSize( (int *) NULL, &ch );
29d87bba 450
d4c99d6f
RR
451 PrepareDC( dc );
452
c801d85f 453 wxRectangle rect;
d4c99d6f
RR
454 rect.x = dc.LogicalToDeviceX( 0 );
455 rect.y = 0;
456 rect.width = 10000;
457 rect.height = ch;
29d87bba 458
c801d85f
KB
459 if (p->m_children.Number() == 1)
460 {
461 rect.y = dc.LogicalToDeviceY( p->m_y );
462 }
463 else
464 {
465 wxNode *node = p->m_children.Member( new_child )->Previous();
466 wxGenericTreeItem* last_child = (wxGenericTreeItem*)node->Data();
467 rect.y = dc.LogicalToDeviceY( last_child->m_y );
edaa81ae 468 }
29d87bba 469
c801d85f 470 AdjustMyScrollbars();
29d87bba 471
c801d85f
KB
472 if (rect.height > 0) Refresh( FALSE, &rect);
473 }
474 else
475 {
476 AdjustMyScrollbars();
29d87bba 477
c801d85f 478 Refresh();
edaa81ae 479 }
29d87bba 480
c801d85f 481 return m_lastId;
edaa81ae 482}
c801d85f 483
74bedbeb 484long wxTreeCtrl::InsertItem( long parent, wxTreeItem &info, long WXUNUSED(insertAfter) )
c801d85f
KB
485{
486 int oldMask = info.m_mask;
c67daf87 487 wxGenericTreeItem *p = (wxGenericTreeItem *) NULL;
c801d85f
KB
488 if (parent == 0)
489 {
490 if (m_anchor) return -1;
491 }
492 else
493 {
494 p = FindItem( parent );
495 if (!p)
496 {
497 printf( "TreeItem not found.\n" );
498 return -1;
edaa81ae
RR
499 }
500 }
c801d85f
KB
501 long ret = 0;
502 if ((info.m_mask & wxTREE_MASK_HANDLE) == 0)
503 {
504 m_lastId++;
505 info.m_itemId = m_lastId;
506 info.m_mask |= wxTREE_MASK_HANDLE;
507 ret = m_lastId;
508 }
509 else
510 {
511 ret = info.m_itemId;
edaa81ae 512 }
29d87bba 513
c801d85f
KB
514 wxClientDC dc(this);
515 wxGenericTreeItem *new_child = new wxGenericTreeItem( p, info, &dc );
29d87bba 516 if (p)
c801d85f
KB
517 p->AddChild( new_child );
518 else
519 m_anchor = new_child;
29d87bba 520
c801d85f
KB
521 if (p)
522 {
523 CalculatePositions();
389cdc7a 524
e2414cbe 525 if (!p->HasChildren()) p->m_hasChildren = TRUE;
29d87bba 526
c801d85f 527 int ch = 0;
c67daf87 528 GetClientSize( (int *) NULL, &ch );
29d87bba 529
d4c99d6f
RR
530 PrepareDC( dc );
531
c801d85f 532 wxRectangle rect;
d4c99d6f
RR
533 rect.x = dc.LogicalToDeviceX( 0 );
534 rect.y = 0;
535 rect.width = 10000;
536 rect.height = ch;
29d87bba 537
c801d85f
KB
538 if (p->m_children.Number() == 1)
539 {
540 rect.y = dc.LogicalToDeviceY( p->m_y );
541 }
542 else
543 {
544 wxNode *node = p->m_children.Member( new_child )->Previous();
545 wxGenericTreeItem* last_child = (wxGenericTreeItem*)node->Data();
546 rect.y = dc.LogicalToDeviceY( last_child->m_y );
edaa81ae 547 }
29d87bba 548
c801d85f 549 AdjustMyScrollbars();
29d87bba 550
c801d85f
KB
551 if (rect.height > 0) Refresh( FALSE, &rect);
552 }
553 else
554 {
555 AdjustMyScrollbars();
29d87bba 556
c801d85f 557 Refresh();
edaa81ae 558 }
29d87bba 559
c801d85f
KB
560 info.m_mask = oldMask;
561 return ret;
edaa81ae 562}
c801d85f 563
74bedbeb 564bool wxTreeCtrl::ExpandItem( long item, int action )
c801d85f
KB
565{
566 wxGenericTreeItem *i = FindItem( item );
74bedbeb
VZ
567 if (!i)
568 return FALSE;
569
c801d85f
KB
570 switch (action)
571 {
572 case wxTREE_EXPAND_EXPAND:
573 {
574 i->SendExpand( this );
575 break;
74bedbeb
VZ
576 }
577
c801d85f
KB
578 case wxTREE_EXPAND_COLLAPSE_RESET:
579 case wxTREE_EXPAND_COLLAPSE:
580 {
581 wxNode *node = i->m_children.First();
582 while (node)
583 {
584 wxGenericTreeItem *child = (wxGenericTreeItem*)node->Data();
74bedbeb
VZ
585 if ( child->IsExpanded() )
586 ExpandItem( child->m_itemId, wxTREE_EXPAND_COLLAPSE );
587 node = node->Next();
edaa81ae 588 }
29d87bba 589
e2414cbe 590 CalculatePositions();
389cdc7a 591
74bedbeb 592 i->SendCollapse( this );
c801d85f 593 break;
74bedbeb
VZ
594 }
595
c801d85f
KB
596 case wxTREE_EXPAND_TOGGLE:
597 {
74bedbeb 598 if ( i->IsExpanded() )
c801d85f 599 ExpandItem( item, wxTREE_EXPAND_COLLAPSE );
74bedbeb
VZ
600 else
601 ExpandItem( item, wxTREE_EXPAND_EXPAND );
602 return TRUE;
603 }
edaa81ae 604 }
74bedbeb 605
d4c99d6f
RR
606 wxClientDC dc(this);
607 PrepareDC(dc);
608
74bedbeb
VZ
609 int cw = 0;
610 int ch = 0;
611 GetClientSize( &cw, &ch );
d4c99d6f 612
74bedbeb 613 wxRect rect;
d4c99d6f 614 rect.x = dc.LogicalToDeviceX( 0 );
74bedbeb 615 rect.width = cw;
74bedbeb 616 rect.y = dc.LogicalToDeviceY( i->m_y );
d4c99d6f 617 rect.height = ch;
74bedbeb
VZ
618 Refresh( TRUE, &rect );
619
620 AdjustMyScrollbars();
621
c801d85f 622 return TRUE;
edaa81ae 623}
c801d85f 624
a32dd690
VZ
625void wxTreeCtrl::DeleteItem( long item )
626{
627 wxGenericTreeItem *pItem = FindItem( item );
1a5a8367 628 wxCHECK_RET( pItem != NULL, _("wxTreeCtrl::DeleteItem: no such pItem.") );
389cdc7a 629
a32dd690 630 pItem->m_parent->m_children.DeleteObject(pItem);
389cdc7a 631
4c681997
RR
632 Refresh();
633}
634
635void wxTreeCtrl::DeleteChildren( long item )
636{
637 wxGenericTreeItem *pItem = FindItem( item );
1a5a8367 638 wxCHECK_RET( pItem != NULL, _("wxTreeCtrl::DeleteChildren: no such pItem.") );
389cdc7a 639
4c681997 640 pItem->m_children.Clear();
389cdc7a 641
a32dd690
VZ
642 Refresh();
643}
644
74bedbeb 645bool wxTreeCtrl::DeleteAllItems()
c801d85f
KB
646{
647 delete m_anchor;
c67daf87 648 m_anchor = (wxGenericTreeItem *) NULL;
c801d85f
KB
649 Refresh();
650 return TRUE;
edaa81ae 651}
c801d85f
KB
652
653bool wxTreeCtrl::GetItem( wxTreeItem &info ) const
654{
655 wxGenericTreeItem *i = FindItem( info.m_itemId );
656 if (!i) return FALSE;
657 i->GetItem( info );
658 return TRUE;
edaa81ae 659}
c801d85f 660
74bedbeb 661long wxTreeCtrl::GetItemData( long item ) const
c801d85f
KB
662{
663 wxGenericTreeItem *i = FindItem( item );
664 if (!i) return 0;
665 return i->m_data;
edaa81ae 666}
c801d85f 667
74bedbeb 668wxString wxTreeCtrl::GetItemText( long item ) const
c801d85f
KB
669{
670 wxGenericTreeItem *i = FindItem( item );
671 if (!i) return "";
672 return i->m_text;
edaa81ae 673}
c801d85f 674
74bedbeb
VZ
675int wxTreeCtrl::GetItemImage(long item) const
676{
677 wxGenericTreeItem *i = FindItem( item );
678 return i == 0 ? -1 : i->GetImage();
679}
680
681long wxTreeCtrl::GetParent( long item ) const
c801d85f
KB
682{
683 wxGenericTreeItem *i = FindItem( item );
684 if (!i) return -1;
685 i = i->m_parent;
686 if (!i) return -1;
687 return i->m_parent->m_itemId;
edaa81ae 688}
c801d85f 689
74bedbeb 690long wxTreeCtrl::GetRootItem() const
c801d85f
KB
691{
692 if (m_anchor) return m_anchor->m_itemId;
693 return -1;
edaa81ae
RR
694}
695
696long wxTreeCtrl::GetChild( long item ) const
697{
698 wxGenericTreeItem *i = FindItem( item );
699 if (!i) return -1;
700 if (!i->HasChildren()) return -1;
701 wxNode *node = i->m_children.First();
702 i = (wxGenericTreeItem *)node->Data();
703 return i->m_itemId;
704}
705
706long wxTreeCtrl::GetNextItem( long item, int code ) const
707{
708 switch (code)
709 {
710 case wxTREE_NEXT_CARET: return GetSelection();
711 case wxTREE_NEXT_CHILD: return GetChild( item );
712 case wxTREE_NEXT_ROOT: return GetRootItem();
713 case wxTREE_NEXT_PARENT: return GetParent( item );
714 case wxTREE_NEXT_NEXT:
715 {
716 wxGenericTreeItem *i = FindItem( item );
717 if (!i) return -1;
718 if (i->m_parent)
719 {
720 wxNode *node = i->m_parent->m_children.Member( i );
721 if (!node) return -1;
722 node = node->Next();
723 if (!node) return -1;
724 i = (wxGenericTreeItem *)node->Data();
725 return i->m_itemId;
726 }
727 break;
728 }
729 case wxTREE_NEXT_PREVIOUS:
730 {
731 wxGenericTreeItem *i = FindItem( item );
732 if (!i) return -1;
733 if (i->m_parent)
734 {
735 wxNode *node = i->m_parent->m_children.Member( i );
736 if (!node) return -1;
737 node = node->Previous();
738 if (!node) return -1;
739 i = (wxGenericTreeItem *)node->Data();
740 return i->m_itemId;
741 }
742 break;
743 }
744 }
745 return -1;
746}
c801d85f 747
74bedbeb 748long wxTreeCtrl::GetSelection() const
c801d85f 749{
389cdc7a 750 return m_current ? m_current->GetItemId() : -1;
edaa81ae 751}
c801d85f 752
edaa81ae 753bool wxTreeCtrl::SelectItem( long itemId )
c801d85f 754{
389cdc7a
VZ
755 wxGenericTreeItem *pItem = FindItem(itemId);
756 if ( !pItem ) {
1a5a8367 757 wxLogDebug(_("Can't select an item %d which doesn't exist."), itemId);
389cdc7a
VZ
758
759 return FALSE;
760 }
761
d37a02d0 762 SelectItem(pItem);
389cdc7a
VZ
763
764 return TRUE;
edaa81ae 765}
c801d85f 766
c614ed58 767void wxTreeCtrl::SelectItem(wxGenericTreeItem *item)
389cdc7a
VZ
768{
769 if (m_current != item)
770 {
771 if (m_current)
772 {
773 m_current->SetHilight( FALSE );
774 RefreshLine( m_current );
edaa81ae 775 }
389cdc7a
VZ
776 m_current = item;
777 m_current->SetHilight( TRUE );
778 RefreshLine( m_current );
779
c614ed58 780 m_current->SendSelected( this );
389cdc7a
VZ
781 }
782}
783
74bedbeb 784bool wxTreeCtrl::ItemHasChildren( long item ) const
c801d85f
KB
785{
786 wxGenericTreeItem *i = FindItem( item );
787 if (!i) return FALSE;
788 return i->m_hasChildren;
edaa81ae 789}
c801d85f 790
74bedbeb 791void wxTreeCtrl::SetIndent( int indent )
c801d85f
KB
792{
793 m_indent = indent;
794 Refresh();
edaa81ae 795}
c801d85f 796
74bedbeb 797int wxTreeCtrl::GetIndent() const
c801d85f
KB
798{
799 return m_indent;
edaa81ae 800}
c801d85f
KB
801
802bool wxTreeCtrl::SetItem( wxTreeItem &info )
803{
804 wxGenericTreeItem *i = FindItem( info.m_itemId );
805 if (!i) return FALSE;
806 wxClientDC dc(this);
807 i->SetItem( info, &dc );
74bedbeb 808 Refresh();
c801d85f 809 return TRUE;
edaa81ae 810}
c801d85f 811
74bedbeb 812bool wxTreeCtrl::SetItemData( long item, long data )
c801d85f
KB
813{
814 wxGenericTreeItem *i = FindItem( item );
815 if (!i) return FALSE;
816 i->m_data = data;
817 return TRUE;
edaa81ae 818}
c801d85f 819
74bedbeb 820bool wxTreeCtrl::SetItemText( long item, const wxString &text )
c801d85f
KB
821{
822 wxGenericTreeItem *i = FindItem( item );
823 if (!i) return FALSE;
824 wxClientDC dc(this);
825 i->SetText( text, &dc );
826 return TRUE;
edaa81ae 827}
c801d85f 828
74bedbeb
VZ
829void wxTreeCtrl::SetItemImage(long item, int image, int imageSel) const
830{
831 wxGenericTreeItem *i = FindItem( item );
edaa81ae
RR
832 if ( i != 0 )
833 {
74bedbeb
VZ
834 i->SetImage(image);
835 i->SetSelectedImage(imageSel);
836 }
837}
838
edaa81ae
RR
839int wxTreeCtrl::GetItemState( long item, long stateMask ) const
840{
841 wxGenericTreeItem *i = FindItem( item );
842 if (!i) return 0;
843 int ret = 0;
844
845 // Currently, an item is always selected _and_ focussed in wxGTK
846
847 if (stateMask & wxTREE_STATE_FOCUSED)
848 if (m_current == i)
849 ret += wxTREE_STATE_FOCUSED;
850
851 if (stateMask & wxTREE_STATE_SELECTED)
852 if (m_current == i)
853 ret += wxTREE_STATE_SELECTED;
854
855 if (stateMask & wxTREE_STATE_EXPANDED)
856 if (i->IsExpanded())
857 ret += wxTREE_STATE_EXPANDED;
858
859 return ret;
860}
861
862bool wxTreeCtrl::SetItemState( long item, long state, long stateMask )
863{
864 wxGenericTreeItem *i = FindItem( item );
865 if (!i) return FALSE;
866
867 if (stateMask & wxTREE_STATE_FOCUSED)
868 {
869 if (state & wxTREE_STATE_FOCUSED)
870 SelectItem( i );
871 }
872
873 if (stateMask & wxTREE_STATE_SELECTED)
874 {
875 if (state & wxTREE_STATE_SELECTED)
876 SelectItem( i );
877 }
878
879 if (stateMask & wxTREE_STATE_EXPANDED)
880 {
881 if (state & wxTREE_STATE_EXPANDED)
882 ExpandItem( i->m_itemId, wxTREE_EXPAND_EXPAND );
883 else
884 ExpandItem( i->m_itemId, wxTREE_EXPAND_COLLAPSE );
885 }
886
887 return TRUE;
888}
889
c801d85f
KB
890long wxTreeCtrl::HitTest( const wxPoint& point, int &flags )
891{
892 flags = 0;
893 if (!m_anchor) return -1;
894 return m_anchor->HitTest( point, flags );
edaa81ae 895}
c801d85f 896
e2414cbe
RR
897wxImageList *wxTreeCtrl::GetImageList( int which ) const
898{
899 if (which == wxIMAGE_LIST_NORMAL) return m_imageList;
900 return m_smallImageList;
edaa81ae 901}
e2414cbe
RR
902
903void wxTreeCtrl::SetImageList( wxImageList *imageList, int which )
904{
905 if (which == wxIMAGE_LIST_NORMAL)
906 {
907 if (m_imageList) delete m_imageList;
908 m_imageList = imageList;
909 }
910 else
911 {
912 if (m_smallImageList) delete m_smallImageList;
913 m_smallImageList = imageList;
edaa81ae
RR
914 }
915}
e2414cbe 916
74bedbeb 917void wxTreeCtrl::AdjustMyScrollbars()
c801d85f
KB
918{
919 if (m_anchor)
920 {
921 int x = 0;
922 int y = 0;
923 m_anchor->GetSize( x, y );
924 y += 2*m_lineHeight;
925 int x_pos = GetScrollPos( wxHORIZONTAL );
926 int y_pos = GetScrollPos( wxVERTICAL );
927 SetScrollbars( 10, 10, x/10, y/10, x_pos, y_pos );
928 }
929 else
930 {
931 SetScrollbars( 0, 0, 0, 0 );
edaa81ae
RR
932 }
933}
c801d85f
KB
934
935void wxTreeCtrl::PaintLevel( wxGenericTreeItem *item, wxPaintDC &dc, int level, int &y )
936{
4c681997 937 int horizX = level*m_indent;
389cdc7a 938
4c681997
RR
939 item->m_x = horizX+33;
940 item->m_y = y-m_lineHeight/3;
941 item->m_height = m_lineHeight;
389cdc7a 942
4c681997
RR
943 item->SetCross( horizX+15, y );
944
c801d85f 945 int oldY = y;
389cdc7a 946
d4c99d6f
RR
947 int exposed_x = dc.LogicalToDeviceX( 0 );
948 int exposed_y = dc.LogicalToDeviceY( item->m_y-2 );
949
950 if (IsExposed( exposed_x, exposed_y, 1000, m_lineHeight+4 ))
c801d85f 951 {
4c681997
RR
952 int startX = horizX;
953 int endX = horizX + 10;
29d87bba 954
4c681997 955 if (!item->HasChildren()) endX += 20;
d4c99d6f 956
389cdc7a 957 dc.DrawLine( startX, y, endX, y );
29d87bba 958
4c681997 959 if (item->HasChildren())
c801d85f
KB
960 {
961 dc.DrawLine( horizX+20, y, horizX+30, y );
962 dc.SetPen( *wxGREY_PEN );
963 dc.DrawRectangle( horizX+10, y-4, 11, 9 );
964 dc.SetPen( *wxBLACK_PEN );
4c681997
RR
965 dc.DrawLine( horizX+13, y, horizX+18, y );
966 if (item->HasPlus())
967 dc.DrawLine( horizX+15, y-2, horizX+15, y+3 );
edaa81ae 968 }
c801d85f 969
4c681997 970 if (item->HasHilight())
c801d85f 971 {
29d87bba 972 dc.SetTextForeground( wxSystemSettings::GetSystemColour( wxSYS_COLOUR_HIGHLIGHTTEXT ) );
29d87bba 973 dc.SetBrush( *m_hilightBrush );
29d87bba 974 long tw, th;
4c681997 975 dc.GetTextExtent( item->m_text, &tw, &th );
e2414cbe 976 if (m_hasFocus)
389cdc7a 977 {
e2414cbe
RR
978 dc.SetPen( *wxBLACK_PEN );
979 dc.DrawRectangle( item->m_x-2, item->m_y-2, tw+4, th+4 );
389cdc7a
VZ
980 }
981 else
982 {
983 dc.SetPen( *wxTRANSPARENT_PEN );
e2414cbe 984 dc.DrawRectangle( item->m_x-2, item->m_y-2, tw+4, th+4 );
389cdc7a 985 }
4c681997 986 dc.DrawText( item->m_text, item->m_x, item->m_y );
29d87bba 987
29d87bba 988 dc.SetPen( *wxBLACK_PEN );
29d87bba 989 dc.SetTextForeground( *wxBLACK );
389cdc7a 990 dc.SetBrush( *wxWHITE_BRUSH );
29d87bba
VZ
991 }
992 else
4c681997 993 {
d4c99d6f 994 dc.SetBrush( *wxWHITE_BRUSH );
389cdc7a 995 dc.SetPen( *wxTRANSPARENT_PEN );
4c681997
RR
996 long tw, th;
997 dc.GetTextExtent( item->m_text, &tw, &th );
998 dc.DrawRectangle( item->m_x-2, item->m_y-2, tw+4, th+4 );
999 dc.DrawText( item->m_text, item->m_x, item->m_y );
1000 dc.SetPen( *wxBLACK_PEN );
edaa81ae
RR
1001 }
1002 }
4c681997
RR
1003
1004 if (item->NumberOfVisibleChildren() == 0) return;
e2414cbe 1005
389cdc7a
VZ
1006 int semiOldY = y;
1007
4c681997
RR
1008 wxNode *node = item->m_children.First();
1009 while (node)
1010 {
1011 wxGenericTreeItem *child = (wxGenericTreeItem *)node->Data();
29d87bba 1012
c801d85f 1013 y += m_lineHeight;
e2414cbe 1014 semiOldY = y;
389cdc7a 1015
4c681997 1016 PaintLevel( child, dc, level+1, y );
389cdc7a 1017
c801d85f 1018 node = node->Next();
edaa81ae 1019 }
4c681997 1020
e2414cbe 1021 dc.DrawLine( horizX+15, oldY+5, horizX+15, semiOldY );
4c681997 1022}
c801d85f
KB
1023
1024void wxTreeCtrl::OnPaint( const wxPaintEvent &WXUNUSED(event) )
1025{
1026 if (!m_anchor) return;
1027
d4c99d6f
RR
1028 wxPaintDC dc(this);
1029 PrepareDC( dc );
29d87bba 1030
d4c99d6f 1031 dc.SetFont( wxSystemSettings::GetSystemFont( wxSYS_SYSTEM_FONT ) );
29d87bba 1032
d4c99d6f
RR
1033 dc.SetPen( m_dottedPen );
1034 m_lineHeight = (int)(dc.GetCharHeight() + 4);
29d87bba 1035
c801d85f 1036 int y = m_lineHeight / 2 + 2;
d4c99d6f 1037 PaintLevel( m_anchor, dc, 0, y );
edaa81ae 1038}
c801d85f
KB
1039
1040void wxTreeCtrl::OnSetFocus( const wxFocusEvent &WXUNUSED(event) )
1041{
1042 m_hasFocus = TRUE;
1043 if (m_current) RefreshLine( m_current );
edaa81ae 1044}
c801d85f
KB
1045
1046void wxTreeCtrl::OnKillFocus( const wxFocusEvent &WXUNUSED(event) )
1047{
1048 m_hasFocus = FALSE;
1049 if (m_current) RefreshLine( m_current );
edaa81ae 1050}
c801d85f
KB
1051
1052void wxTreeCtrl::OnChar( wxKeyEvent &event )
1053{
1054 event.Skip();
edaa81ae 1055}
c801d85f
KB
1056
1057void wxTreeCtrl::OnMouse( const wxMouseEvent &event )
1058{
1059 if (!event.LeftDown() &&
1060 !event.LeftDClick()) return;
29d87bba 1061
c801d85f
KB
1062 wxClientDC dc(this);
1063 PrepareDC(dc);
1064 long x = dc.DeviceToLogicalX( (long)event.GetX() );
1065 long y = dc.DeviceToLogicalY( (long)event.GetY() );
29d87bba 1066
c801d85f
KB
1067 int flag = 0;
1068 long id = HitTest( wxPoint(x,y), flag );
29d87bba
VZ
1069 if (id == -1)
1070 return;
c801d85f 1071 wxGenericTreeItem *item = FindItem( id );
29d87bba 1072
c801d85f
KB
1073 if (!item) return;
1074 if ((flag != wxTREE_HITTEST_ONITEMBUTTON) &&
1075 (flag != wxTREE_HITTEST_ONITEMLABEL)) return;
29d87bba 1076
389cdc7a 1077 SelectItem(item);
29d87bba
VZ
1078
1079 if (event.LeftDClick())
1080 m_current->SendKeyDown( this );
1081
c801d85f
KB
1082 if (flag == wxTREE_HITTEST_ONITEMBUTTON)
1083 {
1084 ExpandItem( item->m_itemId, wxTREE_EXPAND_TOGGLE );
1085 return;
edaa81ae
RR
1086 }
1087}
c801d85f
KB
1088
1089void wxTreeCtrl::CalculateLevel( wxGenericTreeItem *item, wxPaintDC &dc, int level, int &y )
1090{
4c681997 1091 int horizX = level*m_indent;
389cdc7a 1092
4c681997
RR
1093 item->m_x = horizX+33;
1094 item->m_y = y-m_lineHeight/3-2;
1095 item->m_height = m_lineHeight;
1096
1097 if (item->NumberOfVisibleChildren() == 0) return;
389cdc7a 1098
c801d85f
KB
1099 wxNode *node = item->m_children.First();
1100 while (node)
1101 {
1102 wxGenericTreeItem *child = (wxGenericTreeItem *)node->Data();
389cdc7a 1103
c801d85f 1104 y += m_lineHeight;
4c681997 1105 CalculateLevel( child, dc, level+1, y );
29d87bba 1106
c801d85f 1107 node = node->Next();
edaa81ae
RR
1108 }
1109}
c801d85f 1110
74bedbeb 1111void wxTreeCtrl::CalculatePositions()
c801d85f 1112{
29d87bba
VZ
1113 if (!m_anchor)
1114 return;
1115
c801d85f
KB
1116 wxClientDC dc(this);
1117 PrepareDC( dc );
29d87bba 1118
c801d85f 1119 dc.SetFont( wxSystemSettings::GetSystemFont( wxSYS_SYSTEM_FONT ) );
29d87bba 1120
c801d85f
KB
1121 dc.SetPen( m_dottedPen );
1122 m_lineHeight = (int)(dc.GetCharHeight() + 4);
29d87bba 1123
c801d85f
KB
1124 int y = m_lineHeight / 2 + 2;
1125 CalculateLevel( m_anchor, dc, 0, y );
edaa81ae 1126}
c801d85f
KB
1127
1128wxGenericTreeItem *wxTreeCtrl::FindItem( long itemId ) const
1129{
c67daf87 1130 if (!m_anchor) return (wxGenericTreeItem *) NULL;
c801d85f 1131 return m_anchor->FindItem( itemId );
edaa81ae 1132}
c801d85f
KB
1133
1134void wxTreeCtrl::RefreshLine( wxGenericTreeItem *item )
1135{
1136 if (!item) return;
1137 wxClientDC dc(this);
1138 PrepareDC( dc );
1139 wxRect rect;
1140 rect.x = dc.LogicalToDeviceX( item->m_x-2 );
1141 rect.y = dc.LogicalToDeviceY( item->m_y-2 );
1142 rect.width = 1000;
d4c99d6f 1143 rect.height = dc.GetCharHeight()+6;
c801d85f 1144 Refresh( TRUE, &rect );
edaa81ae 1145}
c801d85f
KB
1146
1147
1148