1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/propgrid/property.cpp 
   3 // Purpose:     wxPGProperty and related support classes 
   4 // Author:      Jaakko Salli 
   8 // Copyright:   (c) Jaakko Salli 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // For compilers that support precompilation, includes "wx/wx.h". 
  13 #include "wx/wxprec.h" 
  23     #include "wx/object.h" 
  25     #include "wx/string.h" 
  28     #include "wx/window.h" 
  31     #include "wx/dcmemory.h" 
  34     #include "wx/settings.h" 
  38 #include "wx/propgrid/propgrid.h" 
  41 #define PWC_CHILD_SUMMARY_LIMIT         16 // Maximum number of children summarized in a parent property's 
  44 #define PWC_CHILD_SUMMARY_CHAR_LIMIT    64 // Character limit of summary field when not editing 
  46 #if wxPG_COMPATIBILITY_1_4 
  48 // Used to establish backwards compatiblity 
  49 const char* g_invalidStringContent 
= "@__TOTALLY_INVALID_STRING__@"; 
  53 // ----------------------------------------------------------------------- 
  55 static void wxPGDrawFocusRect( wxDC
& dc
, const wxRect
& rect 
) 
  57 #if defined(__WXMSW__) && !defined(__WXWINCE__) 
  58     // FIXME: Use DrawFocusRect code above (currently it draws solid line 
  59     //   for caption focus but works ok for other stuff). 
  60     //   Also, it seems that this code may not work in future wx versions. 
  61     dc
.SetLogicalFunction(wxINVERT
); 
  63     wxPen 
pen(*wxBLACK
,1,wxDOT
); 
  64     pen
.SetCap(wxCAP_BUTT
); 
  66     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
  68     dc
.DrawRectangle(rect
); 
  70     dc
.SetLogicalFunction(wxCOPY
); 
  72     dc
.SetLogicalFunction(wxINVERT
); 
  74     dc
.SetPen(wxPen(*wxBLACK
,1,wxDOT
)); 
  75     dc
.SetBrush(*wxTRANSPARENT_BRUSH
); 
  77     dc
.DrawRectangle(rect
); 
  79     dc
.SetLogicalFunction(wxCOPY
); 
  83 // ----------------------------------------------------------------------- 
  85 // ----------------------------------------------------------------------- 
  87 wxSize 
wxPGCellRenderer::GetImageSize( const wxPGProperty
* WXUNUSED(property
), 
  89                                        int WXUNUSED(item
) ) const 
  94 void wxPGCellRenderer::DrawText( wxDC
& dc
, const wxRect
& rect
, 
  95                                  int xOffset
, const wxString
& text 
) const 
  98                  rect
.x
+xOffset
+wxPG_XBEFORETEXT
, 
  99                  rect
.y
+((rect
.height
-dc
.GetCharHeight())/2) ); 
 102 void wxPGCellRenderer::DrawEditorValue( wxDC
& dc
, const wxRect
& rect
, 
 103                                         int xOffset
, const wxString
& text
, 
 104                                         wxPGProperty
* property
, 
 105                                         const wxPGEditor
* editor 
) const 
 107     int yOffset 
= ((rect
.height
-dc
.GetCharHeight())/2); 
 114         rect2
.height 
-= yOffset
; 
 115         editor
->DrawValue( dc
, rect2
, property
, text 
); 
 120                      rect
.x
+xOffset
+wxPG_XBEFORETEXT
, 
 125 void wxPGCellRenderer::DrawCaptionSelectionRect( wxDC
& dc
, int x
, int y
, int w
, int h 
) const 
 127     wxRect 
focusRect(x
,y
+((h
-dc
.GetCharHeight())/2),w
,h
); 
 128     wxPGDrawFocusRect(dc
,focusRect
); 
 131 int wxPGCellRenderer::PreDrawCell( wxDC
& dc
, const wxRect
& rect
, const wxPGCell
& cell
, int flags 
) const 
 135     // If possible, use cell colours 
 136     if ( !(flags 
& DontUseCellBgCol
) ) 
 138         dc
.SetPen(cell
.GetBgCol()); 
 139         dc
.SetBrush(cell
.GetBgCol()); 
 142     if ( !(flags 
& DontUseCellFgCol
) ) 
 144         dc
.SetTextForeground(cell
.GetFgCol()); 
 147     // Draw Background, but only if not rendering in control 
 148     // (as control already has rendered correct background). 
 149     if ( !(flags 
& (Control
|ChoicePopup
)) ) 
 150         dc
.DrawRectangle(rect
); 
 152     const wxBitmap
& bmp 
= cell
.GetBitmap(); 
 154         // Do not draw oversized bitmap outside choice popup 
 155          ((flags 
& ChoicePopup
) || bmp
.GetHeight() < rect
.height 
) 
 159                        rect
.x 
+ wxPG_CONTROL_MARGIN 
+ wxCC_CUSTOM_IMAGE_MARGIN1
, 
 160                        rect
.y 
+ wxPG_CUSTOM_IMAGE_SPACINGY
, 
 162         imageWidth 
= bmp
.GetWidth(); 
 168 // ----------------------------------------------------------------------- 
 169 // wxPGDefaultRenderer 
 170 // ----------------------------------------------------------------------- 
 172 void wxPGDefaultRenderer::Render( wxDC
& dc
, const wxRect
& rect
, 
 173                                   const wxPropertyGrid
* propertyGrid
, wxPGProperty
* property
, 
 174                                   int column
, int item
, int flags 
) const 
 176     bool isUnspecified 
= property
->IsValueUnspecified(); 
 178     if ( column 
== 1 && item 
== -1 ) 
 180         int cmnVal 
= property
->GetCommonValue(); 
 184             if ( !isUnspecified 
) 
 185                 DrawText( dc
, rect
, 0, propertyGrid
->GetCommonValueLabel(cmnVal
) ); 
 190     const wxPGEditor
* editor 
= NULL
; 
 191     const wxPGCell
* cell 
= NULL
; 
 195     int preDrawFlags 
= flags
; 
 197     property
->GetDisplayInfo(column
, item
, flags
, &text
, &cell
); 
 199     imageWidth 
= PreDrawCell( dc
, rect
, *cell
, preDrawFlags 
); 
 203         if ( !isUnspecified 
) 
 205             editor 
= property
->GetColumnEditor(column
); 
 207             // Regular property value 
 209             wxSize imageSize 
= propertyGrid
->GetImageSize(property
, item
); 
 211             wxPGPaintData paintdata
; 
 212             paintdata
.m_parent 
= propertyGrid
; 
 213             paintdata
.m_choiceItem 
= item
; 
 215             if ( imageSize
.x 
> 0 ) 
 217                 wxRect 
imageRect(rect
.x 
+ wxPG_CONTROL_MARGIN 
+ wxCC_CUSTOM_IMAGE_MARGIN1
, 
 218                                  rect
.y
+wxPG_CUSTOM_IMAGE_SPACINGY
, 
 219                                  wxPG_CUSTOM_IMAGE_WIDTH
, 
 220                                  rect
.height
-(wxPG_CUSTOM_IMAGE_SPACINGY
*2)); 
 222                 dc
.SetPen( wxPen(propertyGrid
->GetCellTextColour(), 1, wxSOLID
) ); 
 224                 paintdata
.m_drawnWidth 
= imageSize
.x
; 
 225                 paintdata
.m_drawnHeight 
= imageSize
.y
; 
 227                 property
->OnCustomPaint( dc
, imageRect
, paintdata 
); 
 229                 imageWidth 
= paintdata
.m_drawnWidth
; 
 232             text 
= property
->GetValueAsString(); 
 235             if ( propertyGrid
->GetColumnCount() <= 2 ) 
 237                 wxString unitsString 
= property
->GetAttribute(wxPGGlobalVars
->m_strUnits
, wxEmptyString
); 
 238                 if ( unitsString
.length() ) 
 239                     text 
= wxString::Format(wxS("%s %s"), text
.c_str(), unitsString
.c_str() ); 
 243         if ( text
.length() == 0 ) 
 245             // Try to show inline help if no text 
 246             wxVariant vInlineHelp 
= property
->GetAttribute(wxPGGlobalVars
->m_strInlineHelp
); 
 247             if ( !vInlineHelp
.IsNull() ) 
 249                 text 
= vInlineHelp
.GetString(); 
 250                 dc
.SetTextForeground(propertyGrid
->GetCellDisabledTextColour()); 
 255     int imageOffset 
= property
->GetImageOffset(imageWidth
); 
 257     DrawEditorValue( dc
, rect
, imageOffset
, text
, property
, editor 
); 
 259     // active caption gets nice dotted rectangle 
 260     if ( property
->IsCategory() /*&& column == 0*/ ) 
 262         if ( flags 
& Selected 
) 
 264             if ( imageWidth 
> 0 ) 
 266                 imageOffset 
-= DEFAULT_IMAGE_OFFSET_INCREMENT
; 
 267                 imageWidth 
+= wxCC_CUSTOM_IMAGE_MARGIN2 
+ 4; 
 270             DrawCaptionSelectionRect( dc
, 
 271                                       rect
.x
+wxPG_XBEFORETEXT
-wxPG_CAPRECTXMARGIN
+imageOffset
, 
 272                                       rect
.y
-wxPG_CAPRECTYMARGIN
+1, 
 273                                       ((wxPropertyCategory
*)property
)->GetTextExtent(propertyGrid
, 
 274                                                                                      propertyGrid
->GetCaptionFont()) 
 275                                       +(wxPG_CAPRECTXMARGIN
*2), 
 276                                       propertyGrid
->GetFontHeight()+(wxPG_CAPRECTYMARGIN
*2) ); 
 281 wxSize 
wxPGDefaultRenderer::GetImageSize( const wxPGProperty
* property
, 
 285     if ( property 
&& column 
== 1 ) 
 289             wxBitmap
* bmp 
= property
->GetValueImage(); 
 291             if ( bmp 
&& bmp
->Ok() ) 
 292                 return wxSize(bmp
->GetWidth(),bmp
->GetHeight()); 
 298 // ----------------------------------------------------------------------- 
 300 // ----------------------------------------------------------------------- 
 302 wxPGCellData::wxPGCellData() 
 305     m_hasValidText 
= false; 
 308 // ----------------------------------------------------------------------- 
 310 // ----------------------------------------------------------------------- 
 317 wxPGCell::wxPGCell( const wxString
& text
, 
 318                     const wxBitmap
& bitmap
, 
 319                     const wxColour
& fgCol
, 
 320                     const wxColour
& bgCol 
) 
 323     wxPGCellData
* data 
= new wxPGCellData(); 
 326     data
->m_bitmap 
= bitmap
; 
 327     data
->m_fgCol 
= fgCol
; 
 328     data
->m_bgCol 
= bgCol
; 
 329     data
->m_hasValidText 
= true; 
 332 wxObjectRefData 
*wxPGCell::CloneRefData( const wxObjectRefData 
*data 
) const 
 334     wxPGCellData
* c 
= new wxPGCellData(); 
 335     const wxPGCellData
* o 
= (const wxPGCellData
*) data
; 
 336     c
->m_text 
= o
->m_text
; 
 337     c
->m_bitmap 
= o
->m_bitmap
; 
 338     c
->m_fgCol 
= o
->m_fgCol
; 
 339     c
->m_bgCol 
= o
->m_bgCol
; 
 340     c
->m_hasValidText 
= o
->m_hasValidText
; 
 344 void wxPGCell::SetText( const wxString
& text 
) 
 348     GetData()->SetText(text
); 
 351 void wxPGCell::SetBitmap( const wxBitmap
& bitmap 
) 
 355     GetData()->SetBitmap(bitmap
); 
 358 void wxPGCell::SetFgCol( const wxColour
& col 
) 
 362     GetData()->SetFgCol(col
); 
 365 void wxPGCell::SetBgCol( const wxColour
& col 
) 
 369     GetData()->SetBgCol(col
); 
 372 void wxPGCell::MergeFrom( const wxPGCell
& srcCell 
) 
 376     wxPGCellData
* data 
= GetData(); 
 378     if ( srcCell
.HasText() ) 
 379         data
->SetText(srcCell
.GetText()); 
 381     if ( srcCell
.GetFgCol().IsOk() ) 
 382         data
->SetFgCol(srcCell
.GetFgCol()); 
 384     if ( srcCell
.GetBgCol().IsOk() ) 
 385         data
->SetBgCol(srcCell
.GetBgCol()); 
 387     if ( srcCell
.GetBitmap().IsOk() ) 
 388         data
->SetBitmap(srcCell
.GetBitmap()); 
 391 // ----------------------------------------------------------------------- 
 393 // ----------------------------------------------------------------------- 
 395 IMPLEMENT_ABSTRACT_CLASS(wxPGProperty
, wxObject
) 
 397 wxString
* wxPGProperty::sm_wxPG_LABEL 
= NULL
; 
 399 void wxPGProperty::Init() 
 405     m_parentState 
= NULL
; 
 408     m_clientObject 
= NULL
; 
 410     m_customEditor 
= NULL
; 
 414     m_valueBitmap 
= NULL
; 
 416     m_maxLen 
= 0; // infinite maximum length 
 418     m_flags 
= wxPG_PROP_PROPERTY
; 
 426 void wxPGProperty::Init( const wxString
& label
, const wxString
& name 
) 
 428     // We really need to check if &label and &name are NULL pointers 
 429     // (this can if we are called before property grid has been initalized) 
 431     if ( (&label
) != NULL 
&& label 
!= wxPG_LABEL 
) 
 434     if ( (&name
) != NULL 
&& name 
!= wxPG_LABEL 
) 
 437         DoSetName( m_label 
); 
 442 void wxPGProperty::InitAfterAdded( wxPropertyGridPageState
* pageState
, 
 443                                    wxPropertyGrid
* propgrid 
) 
 446     // Called after property has been added to grid or page 
 447     // (so propgrid can be NULL, too). 
 449     wxPGProperty
* parent 
= m_parent
; 
 450     bool parentIsRoot 
= parent
->IsKindOf(CLASSINFO(wxPGRootProperty
)); 
 452     m_parentState 
= pageState
; 
 454 #if wxPG_COMPATIBILITY_1_4 
 455     // Make sure deprecated virtual functions are not implemented 
 456     wxString s 
= GetValueAsString( 0xFFFF ); 
 457     wxASSERT_MSG( s 
== g_invalidStringContent
, 
 458                   "Implement ValueToString() instead of GetValueAsString()" ); 
 461     if ( !parentIsRoot 
&& !parent
->IsCategory() ) 
 463         m_cells 
= parent
->m_cells
; 
 466     // If in hideable adding mode, or if assigned parent is hideable, then 
 467     // make this one hideable. 
 469          ( !parentIsRoot 
&& parent
->HasFlag(wxPG_PROP_HIDDEN
) ) || 
 470          ( propgrid 
&& (propgrid
->HasInternalFlag(wxPG_FL_ADDING_HIDEABLES
)) ) 
 472         SetFlag( wxPG_PROP_HIDDEN 
); 
 474     // Set custom image flag. 
 475     int custImgHeight 
= OnMeasureImage().y
; 
 476     if ( custImgHeight 
< 0 ) 
 478         SetFlag(wxPG_PROP_CUSTOMIMAGE
); 
 481     if ( propgrid 
&& (propgrid
->HasFlag(wxPG_LIMITED_EDITING
)) ) 
 482         SetFlag(wxPG_PROP_NOEDITOR
); 
 484     // Make sure parent has some parental flags 
 485     if ( !parent
->HasFlag(wxPG_PROP_PARENTAL_FLAGS
) ) 
 486         parent
->SetParentalType(wxPG_PROP_MISC_PARENT
); 
 490         // This is not a category. 
 494         unsigned char depth 
= 1; 
 497             depth 
= parent
->m_depth
; 
 498             if ( !parent
->IsCategory() ) 
 502         unsigned char greyDepth 
= depth
; 
 506             wxPropertyCategory
* pc
; 
 508             if ( parent
->IsCategory() ) 
 509                 pc 
= (wxPropertyCategory
* ) parent
; 
 511                 // This conditional compile is necessary to 
 512                 // bypass some compiler bug. 
 513                 pc 
= pageState
->GetPropertyCategory(parent
); 
 516                 greyDepth 
= pc
->GetDepth(); 
 518                 greyDepth 
= parent
->m_depthBgCol
; 
 521         m_depthBgCol 
= greyDepth
; 
 525         // This is a category. 
 528         unsigned char depth 
= 1; 
 531             depth 
= parent
->m_depth 
+ 1; 
 534         m_depthBgCol 
= depth
; 
 538     // Has initial children 
 539     if ( GetChildCount() ) 
 541         // Check parental flags 
 542         wxASSERT_MSG( ((m_flags 
& wxPG_PROP_PARENTAL_FLAGS
) == 
 543                             wxPG_PROP_AGGREGATE
) || 
 544                       ((m_flags 
& wxPG_PROP_PARENTAL_FLAGS
) == 
 545                             wxPG_PROP_MISC_PARENT
), 
 546                       "wxPGProperty parental flags set incorrectly at " 
 549         if ( HasFlag(wxPG_PROP_AGGREGATE
) ) 
 551             // Properties with private children are not expanded by default. 
 554         else if ( propgrid 
&& propgrid
->HasFlag(wxPG_HIDE_MARGIN
) ) 
 556             // ...unless it cannot be expanded by user and therefore must 
 557             // remain visible at all times 
 562         // Prepare children recursively 
 563         for ( unsigned int i
=0; i
<GetChildCount(); i
++ ) 
 565             wxPGProperty
* child 
= Item(i
); 
 566             child
->InitAfterAdded(pageState
, pageState
->GetGrid()); 
 569         if ( propgrid 
&& (propgrid
->GetExtraStyle() & wxPG_EX_AUTO_UNSPECIFIED_VALUES
) ) 
 570             SetFlagRecursively(wxPG_PROP_AUTO_UNSPECIFIED
, true); 
 574 wxPGProperty::wxPGProperty() 
 581 wxPGProperty::wxPGProperty( const wxString
& label
, const wxString
& name 
) 
 588 wxPGProperty::~wxPGProperty() 
 590     delete m_clientObject
; 
 592     Empty();  // this deletes items 
 594     delete m_valueBitmap
; 
 599     // This makes it easier for us to detect dangling pointers 
 604 bool wxPGProperty::IsSomeParent( wxPGProperty
* candidate 
) const 
 606     wxPGProperty
* parent 
= m_parent
; 
 609         if ( parent 
== candidate 
) 
 611         parent 
= parent
->m_parent
; 
 617 wxString 
wxPGProperty::GetName() const 
 619     wxPGProperty
* parent 
= GetParent(); 
 621     if ( !m_name
.length() || !parent 
|| parent
->IsCategory() || parent
->IsRoot() ) 
 624     return m_parent
->GetName() + wxS(".") + m_name
; 
 627 wxPropertyGrid
* wxPGProperty::GetGrid() const 
 629     if ( !m_parentState 
) 
 631     return m_parentState
->GetGrid(); 
 634 int wxPGProperty::Index( const wxPGProperty
* p 
) const 
 636     for ( unsigned int i 
= 0; i
<m_children
.size(); i
++ ) 
 638         if ( p 
== m_children
[i
] ) 
 644 bool wxPGProperty::ValidateValue( wxVariant
& WXUNUSED(value
), wxPGValidationInfo
& WXUNUSED(validationInfo
) ) const 
 649 void wxPGProperty::OnSetValue() 
 653 void wxPGProperty::RefreshChildren () 
 657 void wxPGProperty::OnValidationFailure( wxVariant
& WXUNUSED(pendingValue
) ) 
 661 void wxPGProperty::GetDisplayInfo( unsigned int column
, 
 665                                    const wxPGCell
** pCell 
) 
 667     const wxPGCell
* cell 
= NULL
; 
 669     if ( !(flags 
& wxPGCellRenderer::ChoicePopup
) ) 
 671         // Not painting listi of choice popups, so get text from property 
 672         cell 
= &GetCell(column
); 
 673         if ( cell
->HasText() ) 
 675             *pString 
= cell
->GetText(); 
 680                 *pString 
= GetLabel(); 
 681             else if ( column 
== 1 ) 
 682                 *pString 
= GetDisplayedString(); 
 683             else if ( column 
== 2 ) 
 684                 *pString 
= GetAttribute(wxPGGlobalVars
->m_strUnits
, wxEmptyString
); 
 689         wxASSERT( column 
== 1 ); 
 691         if ( choiceIndex 
!= wxNOT_FOUND 
) 
 693             const wxPGChoiceEntry
& entry 
= m_choices
[choiceIndex
]; 
 694             if ( entry
.GetBitmap().IsOk() || 
 695                  entry
.GetFgCol().IsOk() || 
 696                  entry
.GetBgCol().IsOk() ) 
 698             *pString 
= m_choices
.GetLabel(choiceIndex
); 
 703         cell 
= &GetCell(column
); 
 705     wxASSERT_MSG( cell
->GetData(), 
 706                   wxString::Format("Invalid cell for property %s", 
 707                                    GetName().c_str()) ); 
 713 wxString wxPGProperty::GetColumnText( unsigned int col, int choiceIndex ) const 
 716     if ( col != 1 || choiceIndex == wxNOT_FOUND ) 
 718         const wxPGCell& cell = GetCell(col); 
 719         if ( cell->HasText() ) 
 721             return cell->GetText(); 
 728                 return GetDisplayedString(); 
 730                 return GetAttribute(wxPGGlobalVars->m_strUnits, wxEmptyString); 
 736         return m_choices.GetLabel(choiceIndex); 
 739     return wxEmptyString; 
 743 void wxPGProperty::DoGenerateComposedValue( wxString
& text
, 
 745                                             const wxVariantList
* valueOverrides
, 
 746                                             wxPGHashMapS2S
* childResults 
) const 
 749     int iMax 
= m_children
.size(); 
 755     if ( iMax 
> PWC_CHILD_SUMMARY_LIMIT 
&& 
 756          !(argFlags 
& wxPG_FULL_VALUE
) ) 
 757         iMax 
= PWC_CHILD_SUMMARY_LIMIT
; 
 759     int iMaxMinusOne 
= iMax
-1; 
 761     if ( !IsTextEditable() ) 
 762         argFlags 
|= wxPG_UNEDITABLE_COMPOSITE_FRAGMENT
; 
 764     wxPGProperty
* curChild 
= m_children
[0]; 
 766     bool overridesLeft 
= false; 
 767     wxVariant overrideValue
; 
 768     wxVariantList::const_iterator node
; 
 770     if ( valueOverrides 
) 
 772         node 
= valueOverrides
->begin(); 
 773         if ( node 
!= valueOverrides
->end() ) 
 775             overrideValue 
= *node
; 
 776             overridesLeft 
= true; 
 780     for ( i 
= 0; i 
< iMax
; i
++ ) 
 782         wxVariant childValue
; 
 784         wxString childLabel 
= curChild
->GetLabel(); 
 786         // Check for value override 
 787         if ( overridesLeft 
&& overrideValue
.GetName() == childLabel 
) 
 789             if ( !overrideValue
.IsNull() ) 
 790                 childValue 
= overrideValue
; 
 792                 childValue 
= curChild
->GetValue(); 
 794             if ( node 
!= valueOverrides
->end() ) 
 795                 overrideValue 
= *node
; 
 797                 overridesLeft 
= false; 
 801             childValue 
= curChild
->GetValue(); 
 805         if ( !childValue
.IsNull() ) 
 807             if ( overridesLeft 
&& 
 808                  curChild
->HasFlag(wxPG_PROP_COMPOSED_VALUE
) && 
 809                  childValue
.GetType() == wxPG_VARIANT_TYPE_LIST 
) 
 811                 wxVariantList
& childList 
= childValue
.GetList(); 
 812                 DoGenerateComposedValue(s
, argFlags
|wxPG_COMPOSITE_FRAGMENT
, 
 813                                         &childList
, childResults
); 
 817                 s 
= curChild
->ValueToString(childValue
, 
 818                                             argFlags
|wxPG_COMPOSITE_FRAGMENT
); 
 822         if ( childResults 
&& curChild
->GetChildCount() ) 
 823             (*childResults
)[curChild
->GetName()] = s
; 
 826         if ( (argFlags 
& wxPG_UNEDITABLE_COMPOSITE_FRAGMENT
) && !s
.length() ) 
 829         if ( !curChild
->GetChildCount() || skip 
) 
 832             text 
+= wxS("[") + s 
+ wxS("]"); 
 834         if ( i 
< iMaxMinusOne 
) 
 836             if ( text
.length() > PWC_CHILD_SUMMARY_CHAR_LIMIT 
&& 
 837                  !(argFlags 
& wxPG_EDITABLE_VALUE
) && 
 838                  !(argFlags 
& wxPG_FULL_VALUE
) ) 
 843                 if ( !curChild
->GetChildCount() ) 
 849             curChild 
= m_children
[i
+1]; 
 853     if ( (unsigned int)i 
< m_children
.size() ) 
 855         if ( !text
.EndsWith(wxS("; ")) ) 
 856             text 
+= wxS("; ..."); 
 862 wxString 
wxPGProperty::ValueToString( wxVariant
& WXUNUSED(value
), 
 865     wxCHECK_MSG( GetChildCount() > 0, 
 867                  "If user property does not have any children, it must " 
 868                  "override GetValueAsString" ); 
 870     // FIXME: Currently code below only works if value is actually m_value 
 871     wxASSERT_MSG( argFlags 
& wxPG_VALUE_IS_CURRENT
, 
 872                   "Sorry, currently default wxPGProperty::ValueToString() " 
 873                   "implementation only works if value is m_value." ); 
 876     DoGenerateComposedValue(text
, argFlags
); 
 880 wxString 
wxPGProperty::GetValueAsString( int argFlags 
) const 
 882 #if wxPG_COMPATIBILITY_1_4 
 883     // This is backwards compatibility test 
 884     // That is, to make sure this function is not overridden 
 885     // (instead, ValueToString() should be). 
 886     if ( argFlags 
== 0xFFFF ) 
 888         // Do not override! (for backwards compliancy) 
 889         return g_invalidStringContent
; 
 893     if ( IsValueUnspecified() ) 
 894         return wxEmptyString
; 
 896     if ( m_commonValue 
== -1 ) 
 898         wxVariant 
value(GetValue()); 
 899         return ValueToString(value
, argFlags
|wxPG_VALUE_IS_CURRENT
); 
 903     // Return common value's string representation 
 904     wxPropertyGrid
* pg 
= GetGrid(); 
 905     const wxPGCommonValue
* cv 
= pg
->GetCommonValue(m_commonValue
); 
 907     if ( argFlags 
& wxPG_FULL_VALUE 
) 
 909         return cv
->GetLabel(); 
 911     else if ( argFlags 
& wxPG_EDITABLE_VALUE 
) 
 913         return cv
->GetEditableText(); 
 917         return cv
->GetLabel(); 
 921 wxString 
wxPGProperty::GetValueString( int argFlags 
) const 
 923     return GetValueAsString(argFlags
); 
 926 bool wxPGProperty::IntToValue( wxVariant
& variant
, int number
, int WXUNUSED(argFlags
) ) const 
 928     variant 
= (long)number
; 
 932 // Convert semicolon delimited tokens into child values. 
 933 bool wxPGProperty::StringToValue( wxVariant
& variant
, const wxString
& text
, int argFlags 
) const 
 935     if ( !GetChildCount() ) 
 938     unsigned int curChild 
= 0; 
 940     unsigned int iMax 
= m_children
.size(); 
 942     if ( iMax 
> PWC_CHILD_SUMMARY_LIMIT 
&& 
 943          !(argFlags 
& wxPG_FULL_VALUE
) ) 
 944         iMax 
= PWC_CHILD_SUMMARY_LIMIT
; 
 946     bool changed 
= false; 
 951     // Its best only to add non-empty group items 
 952     bool addOnlyIfNotEmpty 
= false; 
 953     const wxChar delimeter 
= wxS(';'); 
 955     size_t tokenStart 
= 0xFFFFFF; 
 957     wxVariantList temp_list
; 
 958     wxVariant 
list(temp_list
); 
 960     int propagatedFlags 
= argFlags 
& (wxPG_REPORT_ERROR
|wxPG_PROGRAMMATIC_VALUE
); 
 962     wxLogTrace("propgrid", 
 963                wxT(">> %s.StringToValue('%s')"), GetLabel(), text
); 
 965     wxString::const_iterator it 
= text
.begin(); 
 968     if ( it 
!= text
.end() ) 
 975         // How many units we iterate string forward at the end of loop? 
 976         // We need to keep track of this or risk going to negative 
 977         // with it-- operation. 
 978         unsigned int strPosIncrement 
= 1; 
 980         if ( tokenStart 
!= 0xFFFFFF ) 
 983             if ( a 
== delimeter 
|| a 
== 0 ) 
 985                 token 
= text
.substr(tokenStart
,pos
-tokenStart
); 
 987                 size_t len 
= token
.length(); 
 989                 if ( !addOnlyIfNotEmpty 
|| len 
> 0 ) 
 991                     const wxPGProperty
* child 
= Item(curChild
); 
 992                     wxVariant 
variant(child
->GetValue()); 
 993                     wxString childName 
= child
->GetBaseName(); 
 995                     wxLogTrace("propgrid", 
 996                                wxT("token = '%s', child = %s"), 
 999                     // Add only if editable or setting programmatically 
1000                     if ( (argFlags 
& wxPG_PROGRAMMATIC_VALUE
) || 
1001                          !child
->HasFlag(wxPG_PROP_DISABLED
|wxPG_PROP_READONLY
) ) 
1005                             if ( child
->StringToValue(variant
, token
, 
1006                                  propagatedFlags
|wxPG_COMPOSITE_FRAGMENT
) ) 
1008                                 // We really need to set the variant's name 
1009                                 // *after* child->StringToValue() has been 
1010                                 // called, since variant's value may be set by 
1011                                 // assigning another variant into it, which 
1012                                 // then usually causes name to be copied (ie. 
1013                                 // usually cleared) as well. wxBoolProperty 
1014                                 // being case in point with its use of 
1015                                 // wxPGVariant_Bool macro as an optimization. 
1016                                 variant
.SetName(childName
); 
1017                                 list
.Append(variant
); 
1024                             // Empty, becomes unspecified 
1026                             variant
.SetName(childName
); 
1027                             list
.Append(variant
); 
1033                     if ( curChild 
>= iMax 
) 
1037                 tokenStart 
= 0xFFFFFF; 
1042             // Token is not running 
1043             if ( a 
!= wxS(' ') ) 
1046                 addOnlyIfNotEmpty 
= false; 
1048                 // Is this a group of tokens? 
1049                 if ( a 
== wxS('[') ) 
1053                     if ( it 
!= text
.end() ) ++it
; 
1055                     size_t startPos 
= pos
; 
1057                     // Group item - find end 
1058                     while ( it 
!= text
.end() && depth 
> 0 ) 
1064                         if ( a 
== wxS(']') ) 
1066                         else if ( a 
== wxS('[') ) 
1070                     token 
= text
.substr(startPos
,pos
-startPos
-1); 
1072                     if ( !token
.length() ) 
1075                     const wxPGProperty
* child 
= Item(curChild
); 
1077                     wxVariant oldChildValue 
= child
->GetValue(); 
1078                     wxVariant 
variant(oldChildValue
); 
1080                     if ( (argFlags 
& wxPG_PROGRAMMATIC_VALUE
) || 
1081                          !child
->HasFlag(wxPG_PROP_DISABLED
|wxPG_PROP_READONLY
) ) 
1083                         wxString childName 
= child
->GetBaseName(); 
1085                         bool stvRes 
= child
->StringToValue( variant
, token
, 
1087                         if ( stvRes 
|| (variant 
!= oldChildValue
) ) 
1089                             variant
.SetName(childName
); 
1090                             list
.Append(variant
); 
1101                     if ( curChild 
>= iMax 
) 
1104                     addOnlyIfNotEmpty 
= true; 
1106                     tokenStart 
= 0xFFFFFF; 
1112                     if ( a 
== delimeter 
) 
1113                         strPosIncrement 
-= 1; 
1121         it 
+= strPosIncrement
; 
1123         if ( it 
!= text
.end() ) 
1132         pos 
+= strPosIncrement
; 
1141 bool wxPGProperty::SetValueFromString( const wxString
& text
, int argFlags 
) 
1143     wxVariant 
variant(m_value
); 
1144     bool res 
= StringToValue(variant
, text
, argFlags
); 
1150 bool wxPGProperty::SetValueFromInt( long number
, int argFlags 
) 
1152     wxVariant 
variant(m_value
); 
1153     bool res 
= IntToValue(variant
, number
, argFlags
); 
1159 wxSize 
wxPGProperty::OnMeasureImage( int WXUNUSED(item
) ) const 
1161     if ( m_valueBitmap 
) 
1162         return wxSize(m_valueBitmap
->GetWidth(),-1); 
1167 int wxPGProperty::GetImageOffset( int imageWidth 
) const 
1169     int imageOffset 
= 0; 
1173         // Do not increment offset too much for wide images 
1174         if ( imageWidth 
<= (wxPG_CUSTOM_IMAGE_WIDTH
+5) ) 
1175             imageOffset 
= imageWidth 
+ DEFAULT_IMAGE_OFFSET_INCREMENT
; 
1177             imageOffset 
= imageWidth 
+ 1; 
1183 wxPGCellRenderer
* wxPGProperty::GetCellRenderer( int WXUNUSED(column
) ) const 
1185     return wxPGGlobalVars
->m_defaultRenderer
; 
1188 void wxPGProperty::OnCustomPaint( wxDC
& dc
, 
1192     wxBitmap
* bmp 
= m_valueBitmap
; 
1194     wxCHECK_RET( bmp 
&& bmp
->Ok(), wxT("invalid bitmap") ); 
1196     wxCHECK_RET( rect
.x 
>= 0, wxT("unexpected measure call") ); 
1198     dc
.DrawBitmap(*bmp
,rect
.x
,rect
.y
); 
1201 const wxPGEditor
* wxPGProperty::DoGetEditorClass() const 
1203     return wxPGEditor_TextCtrl
; 
1206 // Default extra property event handling - that is, none at all. 
1207 bool wxPGProperty::OnEvent( wxPropertyGrid
*, wxWindow
*, wxEvent
& ) 
1213 void wxPGProperty::SetValue( wxVariant value
, wxVariant
* pList
, int flags 
) 
1215     // If auto unspecified values are not wanted (via window or property style), 
1216     // then get default value instead of wxNullVariant. 
1217     if ( value
.IsNull() && (flags 
& wxPG_SETVAL_BY_USER
) && 
1218          !UsesAutoUnspecified() ) 
1220         value 
= GetDefaultValue(); 
1223     if ( !value
.IsNull() ) 
1225         wxVariant tempListVariant
; 
1228         // List variants are reserved a special purpose 
1229         // as intermediate containers for child values 
1230         // of properties with children. 
1231         if ( value
.GetType() == wxPG_VARIANT_TYPE_LIST 
) 
1234             // However, situation is different for composed string properties 
1235             if ( HasFlag(wxPG_PROP_COMPOSED_VALUE
) ) 
1237                 tempListVariant 
= value
; 
1238                 pList 
= &tempListVariant
; 
1242             AdaptListToValue(value
, &newValue
); 
1244             //wxLogDebug(wxT(">> %s.SetValue() adapted list value to type '%s'"),GetName().c_str(),value.GetType().c_str()); 
1247         if ( HasFlag( wxPG_PROP_AGGREGATE
) ) 
1248             flags 
|= wxPG_SETVAL_AGGREGATED
; 
1250         if ( pList 
&& !pList
->IsNull() ) 
1252             wxASSERT( pList
->GetType() == wxPG_VARIANT_TYPE_LIST 
); 
1253             wxASSERT( GetChildCount() ); 
1254             wxASSERT( !IsCategory() ); 
1256             wxVariantList
& list 
= pList
->GetList(); 
1257             wxVariantList::iterator node
; 
1260             //wxLogDebug(wxT(">> %s.SetValue() pList parsing"),GetName().c_str()); 
1262             // Children in list can be in any order, but we will give hint to 
1263             // GetPropertyByNameWH(). This optimizes for full list parsing. 
1264             for ( node 
= list
.begin(); node 
!= list
.end(); ++node 
) 
1266                 wxVariant
& childValue 
= *((wxVariant
*)*node
); 
1267                 wxPGProperty
* child 
= GetPropertyByNameWH(childValue
.GetName(), i
); 
1270                     //wxLogDebug(wxT("%i: child = %s, childValue.GetType()=%s"),i,child->GetBaseName().c_str(),childValue.GetType().c_str()); 
1271                     if ( childValue
.GetType() == wxPG_VARIANT_TYPE_LIST 
) 
1273                         if ( child
->HasFlag(wxPG_PROP_AGGREGATE
) && !(flags 
& wxPG_SETVAL_AGGREGATED
) ) 
1275                             wxVariant listRefCopy 
= childValue
; 
1276                             child
->SetValue(childValue
, &listRefCopy
, flags
|wxPG_SETVAL_FROM_PARENT
); 
1280                             wxVariant oldVal 
= child
->GetValue(); 
1281                             child
->SetValue(oldVal
, &childValue
, flags
|wxPG_SETVAL_FROM_PARENT
); 
1284                     else if ( child
->GetValue() != childValue 
) 
1286                         // For aggregate properties, we will trust RefreshChildren() 
1287                         // to update child values. 
1288                         if ( !HasFlag(wxPG_PROP_AGGREGATE
) ) 
1289                             child
->SetValue(childValue
, NULL
, flags
|wxPG_SETVAL_FROM_PARENT
); 
1290                         if ( flags 
& wxPG_SETVAL_BY_USER 
) 
1291                             child
->SetFlag(wxPG_PROP_MODIFIED
); 
1298         if ( !value
.IsNull() ) 
1304         if ( flags 
& wxPG_SETVAL_BY_USER 
) 
1305             SetFlag(wxPG_PROP_MODIFIED
); 
1307         if ( HasFlag(wxPG_PROP_AGGREGATE
) ) 
1312         if ( m_commonValue 
!= -1 ) 
1314             wxPropertyGrid
* pg 
= GetGrid(); 
1315             if ( !pg 
|| m_commonValue 
!= pg
->GetUnspecifiedCommonValue() ) 
1321         // Set children to unspecified, but only if aggregate or 
1322         // value is <composed> 
1323         if ( AreChildrenComponents() ) 
1326             for ( i
=0; i
<GetChildCount(); i
++ ) 
1327                 Item(i
)->SetValue(value
, NULL
, flags
|wxPG_SETVAL_FROM_PARENT
); 
1331     if ( !(flags 
& wxPG_SETVAL_FROM_PARENT
) ) 
1332         UpdateParentValues(); 
1335     // Update editor control 
1338     // We need to check for these, otherwise GetGrid() may fail. 
1339     if ( flags 
& wxPG_SETVAL_REFRESH_EDITOR 
) 
1342         wxPropertyGrid
* pg 
= GetGridIfDisplayed(); 
1344             pg
->DrawItemAndValueRelated(this); 
1349 void wxPGProperty::SetValueInEvent( wxVariant value 
) const 
1351     GetGrid()->ValueChangeInEvent(value
); 
1354 void wxPGProperty::SetFlagRecursively( FlagType flag
, bool set 
) 
1362     for ( i 
= 0; i 
< GetChildCount(); i
++ ) 
1363         Item(i
)->SetFlagRecursively(flag
, set
); 
1366 void wxPGProperty::RefreshEditor() 
1371     wxPropertyGrid
* pg 
= GetGrid(); 
1372     if ( pg 
&& pg
->GetSelectedProperty() == this ) 
1373         pg
->RefreshEditor(); 
1376 wxVariant 
wxPGProperty::GetDefaultValue() const 
1378     wxVariant defVal 
= GetAttribute(wxS("DefaultValue")); 
1379     if ( !defVal
.IsNull() ) 
1382     wxVariant value 
= GetValue(); 
1384     if ( !value
.IsNull() ) 
1386         wxString 
valueType(value
.GetType()); 
1388         if ( valueType 
== wxPG_VARIANT_TYPE_LONG 
) 
1389             return wxPGVariant_Zero
; 
1390         if ( valueType 
== wxPG_VARIANT_TYPE_STRING 
) 
1391             return wxPGVariant_EmptyString
; 
1392         if ( valueType 
== wxPG_VARIANT_TYPE_BOOL 
) 
1393             return wxPGVariant_False
; 
1394         if ( valueType 
== wxPG_VARIANT_TYPE_DOUBLE 
) 
1395             return wxVariant(0.0); 
1396         if ( valueType 
== wxPG_VARIANT_TYPE_ARRSTRING 
) 
1397             return wxVariant(wxArrayString()); 
1398         if ( valueType 
== wxS("wxLongLong") ) 
1399             return WXVARIANT(wxLongLong(0)); 
1400         if ( valueType 
== wxS("wxULongLong") ) 
1401             return WXVARIANT(wxULongLong(0)); 
1402         if ( valueType 
== wxS("wxColour") ) 
1403             return WXVARIANT(*wxBLACK
); 
1405         if ( valueType 
== wxPG_VARIANT_TYPE_DATETIME 
) 
1406             return wxVariant(wxDateTime::Now()); 
1408         if ( valueType 
== wxS("wxFont") ) 
1409             return WXVARIANT(*wxNORMAL_FONT
); 
1410         if ( valueType 
== wxS("wxPoint") ) 
1411             return WXVARIANT(wxPoint(0, 0)); 
1412         if ( valueType 
== wxS("wxSize") ) 
1413             return WXVARIANT(wxSize(0, 0)); 
1419 void wxPGProperty::EnsureCells( unsigned int column 
) 
1421     if ( column 
>= m_cells
.size() ) 
1423         // Fill empty slots with default cells 
1424         wxPropertyGrid
* pg 
= GetGrid(); 
1425         wxPGCell defaultCell
; 
1427         if ( !HasFlag(wxPG_PROP_CATEGORY
) ) 
1428             defaultCell 
= pg
->GetPropertyDefaultCell(); 
1430             defaultCell 
= pg
->GetCategoryDefaultCell(); 
1432         // TODO: Replace with resize() call 
1433         unsigned int cellCountMax 
= column
+1; 
1435         for ( unsigned int i
=m_cells
.size(); i
<cellCountMax
; i
++ ) 
1436             m_cells
.push_back(defaultCell
); 
1440 void wxPGProperty::SetCell( int column
, 
1441                             const wxPGCell
& cell 
) 
1443     EnsureCells(column
); 
1445     m_cells
[column
] = cell
; 
1448 void wxPGProperty::AdaptiveSetCell( unsigned int firstCol
, 
1449                                     unsigned int lastCol
, 
1450                                     const wxPGCell
& cell
, 
1451                                     const wxPGCell
& srcData
, 
1452                                     wxPGCellData
* unmodCellData
, 
1453                                     FlagType ignoreWithFlags
, 
1457     // Sets cell in memory optimizing fashion. That is, if 
1458     // current cell data matches unmodCellData, we will 
1459     // simply get reference to data from cell. Otherwise, 
1460     // cell information from srcData is merged into current. 
1463     if ( !(m_flags 
& ignoreWithFlags
) && !IsRoot() ) 
1465         EnsureCells(lastCol
); 
1467         for ( unsigned int col
=firstCol
; col
<=lastCol
; col
++ ) 
1469             if ( m_cells
[col
].GetData() == unmodCellData 
) 
1471                 // Data matches... use cell directly 
1472                 m_cells
[col
] = cell
; 
1476                 // Data did not match... merge valid information 
1477                 m_cells
[col
].MergeFrom(srcData
); 
1484         for ( unsigned int i
=0; i
<GetChildCount(); i
++ ) 
1485             Item(i
)->AdaptiveSetCell( firstCol
, 
1495 const wxPGCell
& wxPGProperty::GetCell( unsigned int column 
) const 
1497     if ( m_cells
.size() > column 
) 
1498         return m_cells
[column
]; 
1500     wxPropertyGrid
* pg 
= GetGrid(); 
1503         return pg
->GetCategoryDefaultCell(); 
1505     return pg
->GetPropertyDefaultCell(); 
1508 wxPGCell
& wxPGProperty::GetCell( unsigned int column 
) 
1510     EnsureCells(column
); 
1511     return m_cells
[column
]; 
1514 void wxPGProperty::SetBackgroundColour( const wxColour
& colour
, 
1517     wxPGProperty
* firstProp 
= this; 
1520     // If category is tried to set recursively, skip it and only 
1521     // affect the children. 
1524         while ( firstProp
->IsCategory() ) 
1526             if ( !firstProp
->GetChildCount() ) 
1528             firstProp 
= firstProp
->Item(0); 
1532     wxPGCell
& firstCell 
= firstProp
->GetCell(0); 
1533     wxPGCellData
* firstCellData 
= firstCell
.GetData(); 
1535     wxPGCell 
newCell(firstCell
); 
1536     newCell
.SetBgCol(colour
); 
1538     srcCell
.SetBgCol(colour
); 
1541                      GetParentState()->GetColumnCount()-1, 
1545                      recursively 
? wxPG_PROP_CATEGORY 
: 0, 
1549 void wxPGProperty::SetTextColour( const wxColour
& colour
, 
1552     wxPGProperty
* firstProp 
= this; 
1555     // If category is tried to set recursively, skip it and only 
1556     // affect the children. 
1559         while ( firstProp
->IsCategory() ) 
1561             if ( !firstProp
->GetChildCount() ) 
1563             firstProp 
= firstProp
->Item(0); 
1567     wxPGCell
& firstCell 
= firstProp
->GetCell(0); 
1568     wxPGCellData
* firstCellData 
= firstCell
.GetData(); 
1570     wxPGCell 
newCell(firstCell
); 
1571     newCell
.SetFgCol(colour
); 
1573     srcCell
.SetFgCol(colour
); 
1576                      GetParentState()->GetColumnCount()-1, 
1580                      recursively 
? wxPG_PROP_CATEGORY 
: 0, 
1584 wxPGEditorDialogAdapter
* wxPGProperty::GetEditorDialog() const 
1589 bool wxPGProperty::DoSetAttribute( const wxString
& WXUNUSED(name
), wxVariant
& WXUNUSED(value
) ) 
1594 void wxPGProperty::SetAttribute( const wxString
& name
, wxVariant value 
) 
1596     if ( DoSetAttribute( name
, value 
) ) 
1598         // Support working without grid, when possible 
1599         if ( wxPGGlobalVars
->HasExtraStyle( wxPG_EX_WRITEONLY_BUILTIN_ATTRIBUTES 
) ) 
1603     m_attributes
.Set( name
, value 
); 
1606 void wxPGProperty::SetAttributes( const wxPGAttributeStorage
& attributes 
) 
1608     wxPGAttributeStorage::const_iterator it 
= attributes
.StartIteration(); 
1611     while ( attributes
.GetNext(it
, variant
) ) 
1612         SetAttribute( variant
.GetName(), variant 
); 
1615 wxVariant 
wxPGProperty::DoGetAttribute( const wxString
& WXUNUSED(name
) ) const 
1621 wxVariant 
wxPGProperty::GetAttribute( const wxString
& name 
) const 
1623     return m_attributes
.FindValue(name
); 
1626 wxString 
wxPGProperty::GetAttribute( const wxString
& name
, const wxString
& defVal 
) const 
1628     wxVariant variant 
= m_attributes
.FindValue(name
); 
1630     if ( !variant
.IsNull() ) 
1631         return variant
.GetString(); 
1636 long wxPGProperty::GetAttributeAsLong( const wxString
& name
, long defVal 
) const 
1638     wxVariant variant 
= m_attributes
.FindValue(name
); 
1640     return wxPGVariantToInt(variant
, defVal
); 
1643 double wxPGProperty::GetAttributeAsDouble( const wxString
& name
, double defVal 
) const 
1646     wxVariant variant 
= m_attributes
.FindValue(name
); 
1648     if ( wxPGVariantToDouble(variant
, &retVal
) ) 
1654 wxVariant 
wxPGProperty::GetAttributesAsList() const 
1656     wxVariantList tempList
; 
1657     wxVariant 
v( tempList
, wxString::Format(wxS("@%s@attr"),m_name
.c_str()) ); 
1659     wxPGAttributeStorage::const_iterator it 
= m_attributes
.StartIteration(); 
1662     while ( m_attributes
.GetNext(it
, variant
) ) 
1668 // Slots of utility flags are NULL 
1669 const unsigned int gs_propFlagToStringSize 
= 14; 
1671 static const wxChar
* gs_propFlagToString
[gs_propFlagToStringSize
] = { 
1688 wxString 
wxPGProperty::GetFlagsAsString( FlagType flagsMask 
) const 
1691     int relevantFlags 
= m_flags 
& flagsMask 
& wxPG_STRING_STORED_FLAGS
; 
1695     for ( i
=0; i
<gs_propFlagToStringSize
; i
++ ) 
1697         if ( relevantFlags 
& a 
) 
1699             const wxChar
* fs 
= gs_propFlagToString
[i
]; 
1711 void wxPGProperty::SetFlagsFromString( const wxString
& str 
) 
1715     WX_PG_TOKENIZER1_BEGIN(str
, wxS('|')) 
1717         for ( i
=0; i
<gs_propFlagToStringSize
; i
++ ) 
1719             const wxChar
* fs 
= gs_propFlagToString
[i
]; 
1720             if ( fs 
&& str 
== fs 
) 
1726     WX_PG_TOKENIZER1_END() 
1728     m_flags 
= (m_flags 
& ~wxPG_STRING_STORED_FLAGS
) | flags
; 
1731 wxValidator
* wxPGProperty::DoGetValidator() const 
1736 int wxPGProperty::InsertChoice( const wxString
& label
, int index
, int value 
) 
1738     wxPropertyGrid
* pg 
= GetGrid(); 
1739     int sel 
= GetChoiceSelection(); 
1743     if ( index 
== wxNOT_FOUND 
) 
1744         index 
= m_choices
.GetCount(); 
1749     m_choices
.Insert(label
, index
, value
); 
1751     if ( sel 
!= newSel 
) 
1752         SetChoiceSelection(newSel
); 
1754     if ( this == pg
->GetSelection() ) 
1755         GetEditorClass()->InsertItem(pg
->GetEditorControl(),label
,index
); 
1761 void wxPGProperty::DeleteChoice( int index 
) 
1763     wxPropertyGrid
* pg 
= GetGrid(); 
1765     int sel 
= GetChoiceSelection(); 
1768     // Adjust current value 
1771         SetValueToUnspecified(); 
1774     else if ( index 
< sel 
) 
1779     m_choices
.RemoveAt(index
); 
1781     if ( sel 
!= newSel 
) 
1782         SetChoiceSelection(newSel
); 
1784     if ( this == pg
->GetSelection() ) 
1785         GetEditorClass()->DeleteItem(pg
->GetEditorControl(), index
); 
1788 int wxPGProperty::GetChoiceSelection() const 
1790     wxVariant value 
= GetValue(); 
1791     wxString valueType 
= value
.GetType(); 
1792     int index 
= wxNOT_FOUND
; 
1794     if ( IsValueUnspecified() || !m_choices
.GetCount() ) 
1797     if ( valueType 
== wxPG_VARIANT_TYPE_LONG 
) 
1799         index 
= value
.GetLong(); 
1801     else if ( valueType 
== wxPG_VARIANT_TYPE_STRING 
) 
1803         index 
= m_choices
.Index(value
.GetString()); 
1805     else if ( valueType 
== wxPG_VARIANT_TYPE_BOOL 
) 
1807         index 
= value
.GetBool()? 1 : 0; 
1813 void wxPGProperty::SetChoiceSelection( int newValue 
) 
1815     // Changes value of a property with choices, but only 
1816     // works if the value type is long or string. 
1817     wxString valueType 
= GetValue().GetType(); 
1819     wxCHECK_RET( m_choices
.IsOk(), wxT("invalid choiceinfo") ); 
1821     if ( valueType 
== wxPG_VARIANT_TYPE_STRING 
) 
1823         SetValue( m_choices
.GetLabel(newValue
) ); 
1825     else  // if ( valueType == wxPG_VARIANT_TYPE_LONG ) 
1827         SetValue( (long) newValue 
); 
1831 bool wxPGProperty::SetChoices( wxPGChoices
& choices 
) 
1833     m_choices
.Assign(choices
); 
1836         // This may be needed to trigger some initialization 
1837         // (but don't do it if property is somewhat uninitialized) 
1838         wxVariant defVal 
= GetDefaultValue(); 
1839         if ( defVal
.IsNull() ) 
1849 const wxPGEditor
* wxPGProperty::GetEditorClass() const 
1851     const wxPGEditor
* editor
; 
1853     if ( !m_customEditor 
) 
1855         editor 
= DoGetEditorClass(); 
1858         editor 
= m_customEditor
; 
1861     // Maybe override editor if common value specified 
1862     if ( GetDisplayedCommonValueCount() ) 
1864         // TextCtrlAndButton -> ComboBoxAndButton 
1865         if ( editor
->IsKindOf(CLASSINFO(wxPGTextCtrlAndButtonEditor
)) ) 
1866             editor 
= wxPGEditor_ChoiceAndButton
; 
1868         // TextCtrl -> ComboBox 
1869         else if ( editor
->IsKindOf(CLASSINFO(wxPGTextCtrlEditor
)) ) 
1870             editor 
= wxPGEditor_ComboBox
; 
1876 bool wxPGProperty::HasVisibleChildren() const 
1880     for ( i
=0; i
<GetChildCount(); i
++ ) 
1882         wxPGProperty
* child 
= Item(i
); 
1884         if ( !child
->HasFlag(wxPG_PROP_HIDDEN
) ) 
1891 bool wxPGProperty::RecreateEditor() 
1893     wxPropertyGrid
* pg 
= GetGrid(); 
1896     wxPGProperty
* selected 
= pg
->GetSelection(); 
1897     if ( this == selected 
) 
1899         pg
->DoSelectProperty(this, wxPG_SEL_FORCE
); 
1906 void wxPGProperty::SetValueImage( wxBitmap
& bmp 
) 
1908     delete m_valueBitmap
; 
1910     if ( &bmp 
&& bmp
.Ok() ) 
1913         wxSize maxSz 
= GetGrid()->GetImageSize(); 
1914         wxSize 
imSz(bmp
.GetWidth(),bmp
.GetHeight()); 
1916         if ( imSz
.y 
!= maxSz
.y 
) 
1918             // Create a memory DC 
1919             wxBitmap
* bmpNew 
= new wxBitmap(maxSz
.x
,maxSz
.y
,bmp
.GetDepth()); 
1922             dc
.SelectObject(*bmpNew
); 
1925             // FIXME: This is ugly - use image or wait for scaling patch. 
1926             double scaleY 
= (double)maxSz
.y 
/ (double)imSz
.y
; 
1928             dc
.SetUserScale(scaleY
, scaleY
); 
1930             dc
.DrawBitmap(bmp
, 0, 0); 
1932             m_valueBitmap 
= bmpNew
; 
1936             m_valueBitmap 
= new wxBitmap(bmp
); 
1939         m_flags 
|= wxPG_PROP_CUSTOMIMAGE
; 
1943         m_valueBitmap 
= NULL
; 
1944         m_flags 
&= ~(wxPG_PROP_CUSTOMIMAGE
); 
1949 wxPGProperty
* wxPGProperty::GetMainParent() const 
1951     const wxPGProperty
* curChild 
= this; 
1952     const wxPGProperty
* curParent 
= m_parent
; 
1954     while ( curParent 
&& !curParent
->IsCategory() ) 
1956         curChild 
= curParent
; 
1957         curParent 
= curParent
->m_parent
; 
1960     return (wxPGProperty
*) curChild
; 
1964 const wxPGProperty
* wxPGProperty::GetLastVisibleSubItem() const 
1967     // Returns last visible sub-item, recursively. 
1968     if ( !IsExpanded() || !GetChildCount() ) 
1971     return Last()->GetLastVisibleSubItem(); 
1975 bool wxPGProperty::IsVisible() const 
1977     const wxPGProperty
* parent
; 
1979     if ( HasFlag(wxPG_PROP_HIDDEN
) ) 
1982     for ( parent 
= GetParent(); parent 
!= NULL
; parent 
= parent
->GetParent() ) 
1984         if ( !parent
->IsExpanded() || parent
->HasFlag(wxPG_PROP_HIDDEN
) ) 
1991 wxPropertyGrid
* wxPGProperty::GetGridIfDisplayed() const 
1993     wxPropertyGridPageState
* state 
= GetParentState(); 
1996     wxPropertyGrid
* propGrid 
= state
->GetGrid(); 
1997     if ( state 
== propGrid
->GetState() ) 
2003 int wxPGProperty::GetY2( int lh 
) const 
2005     const wxPGProperty
* parent
; 
2006     const wxPGProperty
* child 
= this; 
2010     for ( parent 
= GetParent(); parent 
!= NULL
; parent 
= child
->GetParent() ) 
2012         if ( !parent
->IsExpanded() ) 
2014         y 
+= parent
->GetChildrenHeight(lh
, child
->GetIndexInParent()); 
2019     y 
-= lh
;  // need to reduce one level 
2025 int wxPGProperty::GetY() const 
2027     return GetY2(GetGrid()->GetRowHeight()); 
2030 // This is used by Insert etc. 
2031 void wxPGProperty::DoAddChild( wxPGProperty
* prop
, int index
, 
2034     if ( index 
< 0 || (size_t)index 
>= m_children
.size() ) 
2036         if ( correct_mode 
) prop
->m_arrIndex 
= m_children
.size(); 
2037         m_children
.push_back( prop 
); 
2041         m_children
.insert( m_children
.begin()+index
, prop
); 
2042         if ( correct_mode 
) FixIndicesOfChildren( index 
); 
2045     prop
->m_parent 
= this; 
2048 void wxPGProperty::DoPreAddChild( int index
, wxPGProperty
* prop 
) 
2050     wxASSERT_MSG( prop
->GetBaseName().length(), 
2051                   "Property's children must have unique, non-empty " 
2052                   "names within their scope" ); 
2054     prop
->m_arrIndex 
= index
; 
2055     m_children
.insert( m_children
.begin()+index
, 
2058     int custImgHeight 
= prop
->OnMeasureImage().y
; 
2059     if ( custImgHeight 
< 0 /*|| custImgHeight > 1*/ ) 
2060         prop
->m_flags 
|= wxPG_PROP_CUSTOMIMAGE
; 
2062     prop
->m_parent 
= this; 
2065 void wxPGProperty::AddPrivateChild( wxPGProperty
* prop 
) 
2067     if ( !(m_flags 
& wxPG_PROP_PARENTAL_FLAGS
) ) 
2068         SetParentalType(wxPG_PROP_AGGREGATE
); 
2070     wxASSERT_MSG( (m_flags 
& wxPG_PROP_PARENTAL_FLAGS
) == 
2071                     wxPG_PROP_AGGREGATE
, 
2072                   "Do not mix up AddPrivateChild() calls with other " 
2073                   "property adders." ); 
2075     DoPreAddChild( m_children
.size(), prop 
); 
2078 #if wxPG_COMPATIBILITY_1_4 
2079 void wxPGProperty::AddChild( wxPGProperty
* prop 
) 
2081     AddPrivateChild(prop
); 
2085 wxPGProperty
* wxPGProperty::InsertChild( int index
, 
2086                                          wxPGProperty
* childProperty 
) 
2089         index 
= m_children
.size(); 
2091     if ( m_parentState 
) 
2093         m_parentState
->DoInsert(this, index
, childProperty
); 
2097         if ( !(m_flags 
& wxPG_PROP_PARENTAL_FLAGS
) ) 
2098             SetParentalType(wxPG_PROP_MISC_PARENT
); 
2100         wxASSERT_MSG( (m_flags 
& wxPG_PROP_PARENTAL_FLAGS
) == 
2101                         wxPG_PROP_MISC_PARENT
, 
2102                       "Do not mix up AddPrivateChild() calls with other " 
2103                       "property adders." ); 
2105         DoPreAddChild( index
, childProperty 
); 
2108     return childProperty
; 
2111 void wxPGProperty::RemoveChild( wxPGProperty
* p 
) 
2113     wxArrayPGProperty::iterator it
; 
2114     wxArrayPGProperty
& children 
= m_children
; 
2116     for ( it
=children
.begin(); it 
!= children
.end(); it
++ ) 
2120             m_children
.erase(it
); 
2126 void wxPGProperty::AdaptListToValue( wxVariant
& list
, wxVariant
* value 
) const 
2128     wxASSERT( GetChildCount() ); 
2129     wxASSERT( !IsCategory() ); 
2131     *value 
= GetValue(); 
2133     if ( !list
.GetCount() ) 
2136     wxASSERT( GetChildCount() >= (unsigned int)list
.GetCount() ); 
2138     bool allChildrenSpecified
; 
2140     // Don't fully update aggregate properties unless all children have 
2142     if ( HasFlag(wxPG_PROP_AGGREGATE
) ) 
2143         allChildrenSpecified 
= AreAllChildrenSpecified(&list
); 
2145         allChildrenSpecified 
= true; 
2147     wxVariant childValue 
= list
[0]; 
2151     //wxLogDebug(wxT(">> %s.AdaptListToValue()"),GetBaseName().c_str()); 
2153     for ( i
=0; i
<GetChildCount(); i
++ ) 
2155         const wxPGProperty
* child 
= Item(i
); 
2157         if ( childValue
.GetName() == child
->GetBaseName() ) 
2159             //wxLogDebug(wxT("  %s(n=%i), %s"),childValue.GetName().c_str(),n,childValue.GetType().c_str()); 
2161             if ( childValue
.GetType() == wxPG_VARIANT_TYPE_LIST 
) 
2163                 wxVariant 
cv2(child
->GetValue()); 
2164                 child
->AdaptListToValue(childValue
, &cv2
); 
2168             if ( allChildrenSpecified 
) 
2169                 ChildChanged(*value
, i
, childValue
); 
2171             if ( n 
== (unsigned int)list
.GetCount() ) 
2173             childValue 
= list
[n
]; 
2179 void wxPGProperty::FixIndicesOfChildren( unsigned int starthere 
) 
2182     for ( i
=starthere
;i
<GetChildCount();i
++) 
2183         Item(i
)->m_arrIndex 
= i
; 
2187 // Returns (direct) child property with given name (or NULL if not found) 
2188 wxPGProperty
* wxPGProperty::GetPropertyByName( const wxString
& name 
) const 
2192     for ( i
=0; i
<GetChildCount(); i
++ ) 
2194         wxPGProperty
* p 
= Item(i
); 
2195         if ( p
->m_name 
== name 
) 
2199     // Does it have point, then? 
2200     int pos 
= name
.Find(wxS('.')); 
2204     wxPGProperty
* p 
= GetPropertyByName(name
. substr(0,pos
)); 
2206     if ( !p 
|| !p
->GetChildCount() ) 
2209     return p
->GetPropertyByName(name
.substr(pos
+1,name
.length()-pos
-1)); 
2212 wxPGProperty
* wxPGProperty::GetPropertyByNameWH( const wxString
& name
, unsigned int hintIndex 
) const 
2214     unsigned int i 
= hintIndex
; 
2216     if ( i 
>= GetChildCount() ) 
2219     unsigned int lastIndex 
= i 
- 1; 
2221     if ( lastIndex 
>= GetChildCount() ) 
2222         lastIndex 
= GetChildCount() - 1; 
2226         wxPGProperty
* p 
= Item(i
); 
2227         if ( p
->m_name 
== name 
) 
2230         if ( i 
== lastIndex 
) 
2234         if ( i 
== GetChildCount() ) 
2241 int wxPGProperty::GetChildrenHeight( int lh
, int iMax_ 
) const 
2243     // Returns height of children, recursively, and 
2244     // by taking expanded/collapsed status into account. 
2246     // iMax is used when finding property y-positions. 
2252         iMax_ 
= GetChildCount(); 
2254     unsigned int iMax 
= iMax_
; 
2256     wxASSERT( iMax 
<= GetChildCount() ); 
2258     if ( !IsExpanded() && GetParent() ) 
2263         wxPGProperty
* pwc 
= (wxPGProperty
*) Item(i
); 
2265         if ( !pwc
->HasFlag(wxPG_PROP_HIDDEN
) ) 
2267             if ( !pwc
->IsExpanded() || 
2268                  pwc
->GetChildCount() == 0 ) 
2271                 h 
+= pwc
->GetChildrenHeight(lh
) + lh
; 
2280 wxPGProperty
* wxPGProperty::GetItemAtY( unsigned int y
, unsigned int lh
, unsigned int* nextItemY 
) const 
2282     wxASSERT( nextItemY 
); 
2284     // Linear search at the moment 
2286     // nextItemY = y of next visible property, final value will be written back. 
2287     wxPGProperty
* result 
= NULL
; 
2288     wxPGProperty
* current 
= NULL
; 
2289     unsigned int iy 
= *nextItemY
; 
2291     unsigned int iMax 
= GetChildCount(); 
2295         wxPGProperty
* pwc 
= Item(i
); 
2297         if ( !pwc
->HasFlag(wxPG_PROP_HIDDEN
) ) 
2308             if ( pwc
->IsExpanded() && 
2309                  pwc
->GetChildCount() > 0 ) 
2311                 result 
= (wxPGProperty
*) pwc
->GetItemAtY( y
, lh
, &iy 
); 
2323     if ( !result 
&& y 
< iy 
) 
2330         wxLogDebug(wxT("%s::GetItemAtY(%i) -> %s"),this->GetLabel().c_str(),y,current->GetLabel().c_str()); 
2332         wxLogDebug(wxT("%s::GetItemAtY(%i) -> NULL"),this->GetLabel().c_str(),y); 
2335     return (wxPGProperty
*) result
; 
2338 void wxPGProperty::Empty() 
2341     if ( !HasFlag(wxPG_PROP_CHILDREN_ARE_COPIES
) ) 
2343         for ( i
=0; i
<GetChildCount(); i
++ ) 
2345             delete m_children
[i
]; 
2352 void wxPGProperty::DeleteChildren() 
2354     wxPropertyGridPageState
* state 
= m_parentState
; 
2356     while ( GetChildCount() ) 
2358         wxPGProperty
* child 
= Item(GetChildCount()-1); 
2359         state
->DoDelete(child
, true); 
2363 void wxPGProperty::ChildChanged( wxVariant
& WXUNUSED(thisValue
), 
2364                                  int WXUNUSED(childIndex
), 
2365                                  wxVariant
& WXUNUSED(childValue
) ) const 
2369 bool wxPGProperty::AreAllChildrenSpecified( wxVariant
* pendingList 
) const 
2373     const wxVariantList
* pList 
= NULL
; 
2374     wxVariantList::const_iterator node
; 
2378         pList 
= &pendingList
->GetList(); 
2379         node 
= pList
->begin(); 
2382     for ( i
=0; i
<GetChildCount(); i
++ ) 
2384         wxPGProperty
* child 
= Item(i
); 
2385         const wxVariant
* listValue 
= NULL
; 
2390             const wxString
& childName 
= child
->GetBaseName(); 
2392             for ( ; node 
!= pList
->end(); ++node 
) 
2394                 const wxVariant
& item 
= *((const wxVariant
*)*node
); 
2395                 if ( item
.GetName() == childName 
) 
2405             value 
= child
->GetValue(); 
2407         if ( value
.IsNull() ) 
2410         // Check recursively 
2411         if ( child
->GetChildCount() ) 
2413             const wxVariant
* childList 
= NULL
; 
2415             if ( listValue 
&& listValue
->GetType() == wxPG_VARIANT_TYPE_LIST 
) 
2416                 childList 
= listValue
; 
2418             if ( !child
->AreAllChildrenSpecified((wxVariant
*)childList
) ) 
2426 wxPGProperty
* wxPGProperty::UpdateParentValues() 
2428     wxPGProperty
* parent 
= m_parent
; 
2429     if ( parent 
&& parent
->HasFlag(wxPG_PROP_COMPOSED_VALUE
) && 
2430          !parent
->IsCategory() && !parent
->IsRoot() ) 
2433         parent
->DoGenerateComposedValue(s
); 
2434         parent
->m_value 
= s
; 
2435         return parent
->UpdateParentValues(); 
2440 bool wxPGProperty::IsTextEditable() const 
2442     if ( HasFlag(wxPG_PROP_READONLY
) ) 
2445     if ( HasFlag(wxPG_PROP_NOEDITOR
) && 
2447           wxString(GetEditorClass()->GetClassInfo()->GetClassName()).EndsWith(wxS("Button"))) 
2454 // Call after fixed sub-properties added/removed after creation. 
2455 // if oldSelInd >= 0 and < new max items, then selection is 
2456 // moved to it. Note: oldSelInd -2 indicates that this property 
2457 // should be selected. 
2458 void wxPGProperty::SubPropsChanged( int oldSelInd 
) 
2460     wxPropertyGridPageState
* state 
= GetParentState(); 
2461     wxPropertyGrid
* grid 
= state
->GetGrid(); 
2464     // Re-repare children (recursively) 
2465     for ( unsigned int i
=0; i
<GetChildCount(); i
++ ) 
2467         wxPGProperty
* child 
= Item(i
); 
2468         child
->InitAfterAdded(state
, grid
); 
2471     wxPGProperty
* sel 
= NULL
; 
2472     if ( oldSelInd 
>= (int)m_children
.size() ) 
2473         oldSelInd 
= (int)m_children
.size() - 1; 
2475     if ( oldSelInd 
>= 0 ) 
2476         sel 
= m_children
[oldSelInd
]; 
2477     else if ( oldSelInd 
== -2 ) 
2481         state
->DoSelectProperty(sel
); 
2483     if ( state 
== grid
->GetState() ) 
2485         grid
->GetPanel()->Refresh(); 
2489 // ----------------------------------------------------------------------- 
2491 // ----------------------------------------------------------------------- 
2493 WX_PG_IMPLEMENT_PROPERTY_CLASS_PLAIN(wxPGRootProperty
,none
,TextCtrl
) 
2494 IMPLEMENT_DYNAMIC_CLASS(wxPGRootProperty
, wxPGProperty
) 
2497 wxPGRootProperty::wxPGRootProperty( const wxString
& name 
) 
2507 wxPGRootProperty::~wxPGRootProperty() 
2512 // ----------------------------------------------------------------------- 
2513 // wxPropertyCategory 
2514 // ----------------------------------------------------------------------- 
2516 WX_PG_IMPLEMENT_PROPERTY_CLASS_PLAIN(wxPropertyCategory
,none
,TextCtrl
) 
2517 IMPLEMENT_DYNAMIC_CLASS(wxPropertyCategory
, wxPGProperty
) 
2519 void wxPropertyCategory::Init() 
2521     // don't set colour - prepareadditem method should do this 
2522     SetParentalType(wxPG_PROP_CATEGORY
); 
2523     m_capFgColIndex 
= 1; 
2527 wxPropertyCategory::wxPropertyCategory() 
2534 wxPropertyCategory::wxPropertyCategory( const wxString 
&label
, const wxString
& name 
) 
2535     : wxPGProperty(label
,name
) 
2541 wxPropertyCategory::~wxPropertyCategory() 
2546 wxString 
wxPropertyCategory::ValueToString( wxVariant
& WXUNUSED(value
), 
2547                                             int WXUNUSED(argFlags
) ) const 
2549     return wxEmptyString
; 
2552 int wxPropertyCategory::GetTextExtent( const wxWindow
* wnd
, const wxFont
& font 
) const 
2554     if ( m_textExtent 
> 0 ) 
2555         return m_textExtent
; 
2557         ((wxWindow
*)wnd
)->GetTextExtent( m_label
, &x
, &y
, 0, 0, &font 
); 
2561 void wxPropertyCategory::CalculateTextExtent( wxWindow
* wnd
, const wxFont
& font 
) 
2564         wnd
->GetTextExtent( m_label
, &x
, &y
, 0, 0, &font 
); 
2568 // ----------------------------------------------------------------------- 
2570 // ----------------------------------------------------------------------- 
2572 wxPGChoiceEntry
& wxPGChoices::Add( const wxString
& label
, int value 
) 
2576     wxPGChoiceEntry 
entry(label
, value
); 
2577     return m_data
->Insert( -1, entry 
); 
2580 // ----------------------------------------------------------------------- 
2582 wxPGChoiceEntry
& wxPGChoices::Add( const wxString
& label
, const wxBitmap
& bitmap
, int value 
) 
2586     wxPGChoiceEntry 
entry(label
, value
); 
2587     entry
.SetBitmap(bitmap
); 
2588     return m_data
->Insert( -1, entry 
); 
2591 // ----------------------------------------------------------------------- 
2593 wxPGChoiceEntry
& wxPGChoices::Insert( const wxPGChoiceEntry
& entry
, int index 
) 
2597     return m_data
->Insert( index
, entry 
); 
2600 // ----------------------------------------------------------------------- 
2602 wxPGChoiceEntry
& wxPGChoices::Insert( const wxString
& label
, int index
, int value 
) 
2606     wxPGChoiceEntry 
entry(label
, value
); 
2607     return m_data
->Insert( index
, entry 
); 
2610 // ----------------------------------------------------------------------- 
2612 wxPGChoiceEntry
& wxPGChoices::AddAsSorted( const wxString
& label
, int value 
) 
2618     while ( index 
< GetCount() ) 
2620         int cmpRes 
= GetLabel(index
).Cmp(label
); 
2626     wxPGChoiceEntry 
entry(label
, value
); 
2627     return m_data
->Insert( index
, entry 
); 
2630 // ----------------------------------------------------------------------- 
2632 void wxPGChoices::Add( const wxChar
** labels
, const ValArrItem
* values 
) 
2636     unsigned int itemcount 
= 0; 
2637     const wxChar
** p 
= &labels
[0]; 
2638     while ( *p 
) { p
++; itemcount
++; } 
2641     for ( i 
= 0; i 
< itemcount
; i
++ ) 
2646         wxPGChoiceEntry 
entry(labels
[i
], value
); 
2647         m_data
->Insert( i
, entry 
); 
2651 // ----------------------------------------------------------------------- 
2653 void wxPGChoices::Add( const wxArrayString
& arr
, const wxArrayInt
& arrint 
) 
2658     unsigned int itemcount 
= arr
.size(); 
2660     for ( i 
= 0; i 
< itemcount
; i
++ ) 
2663         if ( &arrint 
&& arrint
.size() ) 
2665         wxPGChoiceEntry 
entry(arr
[i
], value
); 
2666         m_data
->Insert( i
, entry 
); 
2670 // ----------------------------------------------------------------------- 
2672 void wxPGChoices::RemoveAt(size_t nIndex
, size_t count
) 
2676     wxASSERT( m_data
->m_refCount 
!= 0xFFFFFFF ); 
2677     m_data
->m_items
.erase(m_data
->m_items
.begin()+nIndex
, 
2678                           m_data
->m_items
.begin()+nIndex
+count
); 
2681 // ----------------------------------------------------------------------- 
2683 void wxPGChoices::Clear() 
2685     if ( m_data 
!= wxPGChoicesEmptyData 
) 
2692 // ----------------------------------------------------------------------- 
2694 int wxPGChoices::Index( const wxString
& str 
) const 
2699         for ( i
=0; i
< m_data
->GetCount(); i
++ ) 
2701             const wxPGChoiceEntry
& entry 
= m_data
->Item(i
); 
2702             if ( entry
.HasText() && entry
.GetText() == str 
) 
2709 // ----------------------------------------------------------------------- 
2711 int wxPGChoices::Index( int val 
) const 
2716         for ( i
=0; i
< m_data
->GetCount(); i
++ ) 
2718             const wxPGChoiceEntry
& entry 
= m_data
->Item(i
); 
2719             if ( entry
.GetValue() == val 
) 
2726 // ----------------------------------------------------------------------- 
2728 wxArrayString 
wxPGChoices::GetLabels() const 
2733     if ( this && IsOk() ) 
2734         for ( i
=0; i
<GetCount(); i
++ ) 
2735             arr
.push_back(GetLabel(i
)); 
2740 // ----------------------------------------------------------------------- 
2742 wxArrayInt 
wxPGChoices::GetValuesForStrings( const wxArrayString
& strings 
) const 
2749         for ( i
=0; i
< strings
.size(); i
++ ) 
2751             int index 
= Index(strings
[i
]); 
2753                 arr
.Add(GetValue(index
)); 
2755                 arr
.Add(wxPG_INVALID_VALUE
); 
2762 // ----------------------------------------------------------------------- 
2764 wxArrayInt 
wxPGChoices::GetIndicesForStrings( const wxArrayString
& strings
, 
2765                                               wxArrayString
* unmatched 
) const 
2772         for ( i
=0; i
< strings
.size(); i
++ ) 
2774             const wxString
& str 
= strings
[i
]; 
2775             int index 
= Index(str
); 
2778             else if ( unmatched 
) 
2779                 unmatched
->Add(str
); 
2786 // ----------------------------------------------------------------------- 
2788 void wxPGChoices::AllocExclusive() 
2792     if ( m_data
->m_refCount 
!= 1 ) 
2794         wxPGChoicesData
* data 
= new wxPGChoicesData(); 
2795         data
->CopyDataFrom(m_data
); 
2801 // ----------------------------------------------------------------------- 
2803 void wxPGChoices::AssignData( wxPGChoicesData
* data 
) 
2807     if ( data 
!= wxPGChoicesEmptyData 
) 
2814 // ----------------------------------------------------------------------- 
2816 void wxPGChoices::Init() 
2818     m_data 
= wxPGChoicesEmptyData
; 
2821 // ----------------------------------------------------------------------- 
2823 void wxPGChoices::Free() 
2825     if ( m_data 
!= wxPGChoicesEmptyData 
) 
2828         m_data 
= wxPGChoicesEmptyData
; 
2832 // ----------------------------------------------------------------------- 
2833 // wxPGAttributeStorage 
2834 // ----------------------------------------------------------------------- 
2836 wxPGAttributeStorage::wxPGAttributeStorage() 
2840 wxPGAttributeStorage::~wxPGAttributeStorage() 
2842     wxPGHashMapS2P::iterator it
; 
2844     for ( it 
= m_map
.begin(); it 
!= m_map
.end(); ++it 
) 
2846         wxVariantData
* data 
= (wxVariantData
*) it
->second
; 
2851 void wxPGAttributeStorage::Set( const wxString
& name
, const wxVariant
& value 
) 
2853     wxVariantData
* data 
= value
.GetData(); 
2856     wxPGHashMapS2P::iterator it 
= m_map
.find(name
); 
2857     if ( it 
!= m_map
.end() ) 
2859         ((wxVariantData
*)it
->second
)->DecRef(); 
2863             // If Null variant, just remove from set 
2877 #endif  // wxUSE_PROPGRID