]> git.saurik.com Git - wxWidgets.git/blob - utils/configtool/src/configitem.cpp
Added first cut wxWindows Configuration Tool
[wxWidgets.git] / utils / configtool / src / configitem.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: configitem.cpp
3 // Purpose: wxWindows Configuration Tool config item class
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 2003-06-03
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart
9 // Licence:
10 /////////////////////////////////////////////////////////////////////////////
11
12 #ifdef __GNUG__
13 #pragma implementation "configitem.h"
14 #endif
15
16 // Includes other headers for precompiled compilation
17 #include "wx/wx.h"
18
19 #ifdef __BORLANDC__
20 #pragma hdrstop
21 #endif
22
23 #include "wx/tokenzr.h"
24
25 #include "configitem.h"
26 #include "configtree.h"
27 #include "configtooldoc.h"
28 #include "configtoolview.h"
29 #include "wxconfigtool.h"
30 #include "mainframe.h"
31
32 IMPLEMENT_CLASS(ctConfigItem, wxObject)
33
34 ctConfigItem::ctConfigItem()
35 {
36 m_modified = FALSE;
37 m_type = ctTypeBoolCheck;
38 m_treeItemId = wxTreeItemId();
39 m_enabled = TRUE;
40 m_parent = NULL;
41 m_active = TRUE;
42 }
43
44 ctConfigItem::ctConfigItem(ctConfigItem* parent, ctConfigType type, const wxString& name)
45 {
46 m_modified = FALSE;
47 m_type = type;
48 m_treeItemId = wxTreeItemId();
49 m_enabled = FALSE;
50 m_active = TRUE;
51 SetName(name);
52 m_parent = parent;
53 if (parent)
54 parent->AddChild(this);
55 }
56
57 ctConfigItem::~ctConfigItem()
58 {
59 ctConfigTreeCtrl* treeCtrl = wxGetApp().GetMainFrame()->GetConfigTreeCtrl();
60 if (m_treeItemId.IsOk() && treeCtrl)
61 {
62 ctTreeItemData* data = (ctTreeItemData*) treeCtrl->GetItemData(m_treeItemId);
63 if (data)
64 data->SetConfigItem(NULL);
65 }
66 if (GetParent())
67 GetParent()->RemoveChild(this);
68 else
69 {
70 if (wxGetApp().GetMainFrame()->GetDocument() &&
71 wxGetApp().GetMainFrame()->GetDocument()->GetTopItem() == this)
72 wxGetApp().GetMainFrame()->GetDocument()->SetTopItem(NULL);
73 }
74
75 Clear();
76 }
77
78 /// Can we edit this property?
79 bool ctConfigItem::CanEditProperty(const wxString& propName) const
80 {
81 ctProperty* prop = m_properties.FindProperty(propName);
82 if (prop)
83 return !prop->GetReadOnly();
84 else
85 return FALSE;
86 }
87
88 /// Assignment operator.
89 void ctConfigItem::operator= (const ctConfigItem& item)
90 {
91 m_properties = item.m_properties;
92 m_modified = item.m_modified;
93 m_defaultProperty = item.m_defaultProperty;
94 m_type = item.m_type;
95 m_enabled = item.m_enabled;
96 m_active = item.m_active;
97 }
98
99 /// Sets the name property.
100 void ctConfigItem::SetName(const wxString& name )
101 {
102 m_properties.SetProperty(wxT("name"), name);
103 }
104
105 /// Clear children
106 void ctConfigItem::Clear()
107 {
108 wxNode* node = m_children.GetFirst();
109 while (node)
110 {
111 wxNode* next = node->GetNext();
112 ctConfigItem* child = (ctConfigItem*) node->GetData();
113
114 // This should delete 'node' too, assuming
115 // child's m_parent points to 'this'. If not,
116 // it'll be cleaned up by m_children.Clear().
117 delete child;
118
119 node = next;
120 }
121 m_children.Clear();
122 }
123
124 // Get the nth child
125 ctConfigItem* ctConfigItem::GetChild(int n) const
126 {
127 wxASSERT ( n < GetChildCount() && n > -1 );
128
129 if ( n < GetChildCount() && n > -1 )
130 {
131 ctConfigItem* child = wxDynamicCast(m_children.Nth(n)->Data(), ctConfigItem);
132 return child;
133 }
134 else
135 return NULL;
136 }
137
138 // Get the child count
139 int ctConfigItem::GetChildCount() const
140 {
141 return m_children.GetCount();
142 }
143
144 /// Add a child
145 void ctConfigItem::AddChild(ctConfigItem* item)
146 {
147 m_children.Append(item);
148 item->SetParent(this);
149 }
150
151 /// Remove (but don't delete) a child
152 void ctConfigItem::RemoveChild(ctConfigItem* item)
153 {
154 m_children.DeleteObject(item);
155 item->SetParent(NULL);
156 }
157
158 /// Initialise standard properties
159 void ctConfigItem::InitProperties()
160 {
161 ctProperty* prop = m_properties.FindProperty(wxT("name"));
162 if (!prop)
163 {
164 prop = new ctProperty;
165 m_properties.AddProperty(prop);
166 }
167 prop->SetDescription(_("<B>Name</B><P> The name of the configuration setting."));
168 prop->SetReadOnly(TRUE);
169
170 m_properties.AddProperty(
171 new ctProperty(
172 wxT("<B>Description</B><P> The setting description."),
173 wxVariant(wxT(""), wxT("description")),
174 wxT("multiline")));
175
176 m_properties.AddProperty(
177 new ctProperty(
178 wxT("<B>Default-state</B><P> The default state."),
179 wxVariant((bool) TRUE, wxT("default-state")),
180 wxT("bool")));
181
182 if (GetType() == ctTypeString)
183 {
184 m_properties.AddProperty(
185 new ctProperty(
186 wxT("<B>Default-value</B><P> The default value."),
187 wxVariant((bool) TRUE, wxT("default-value")),
188 wxT("")));
189 }
190 else if (GetType() == ctTypeInteger)
191 {
192 m_properties.AddProperty(
193 new ctProperty(
194 wxT("<B>Default-value</B><P> The default value."),
195 wxVariant((long) 0, wxT("default-value")),
196 wxT("")));
197 }
198
199 m_properties.AddProperty(
200 new ctProperty(
201 wxT("<B>Requires</B><P> When any of the given settings are 0, this setting <I>must</I> be 0. Taking wxUSE_ZIPSTREAM as an example:<P> If wxUSE_STREAMS is 0, then wxUSE_ZIPSTREAM must be 0.<BR>If wxUSE_STREAMS is 1, then wxUSE_ZIPSTREAM may be 0 or 1."),
202 wxVariant(wxT(""), wxT("requires")),
203 wxT("configitems")));
204
205 m_properties.AddProperty(
206 new ctProperty(
207 wxT("<B>Precludes</B><P> When any of these settings are 1, this setting <I>must</I> be 0. Taking wxUSE_ODBC as an example:<P> If wxUSE_UNICODE is 1, then wxUSE_ODBC must be 0.<BR>If wxUSE_UNICODE is 0, then wxUSE_ODBC may be 0 or 1."),
208 wxVariant(wxT(""), wxT("precludes")),
209 wxT("configitems")));
210
211 m_properties.AddProperty(
212 new ctProperty(
213 wxT("<B>Enabled-if</B><P> When any of these settings are 1, this setting <I>must</I> be 1."),
214 wxVariant(wxT(""), wxT("enabled-if")),
215 wxT("configitems")));
216
217 m_properties.AddProperty(
218 new ctProperty(
219 wxT("<B>Enabled-if-not</B><P> When any of these settings are 0, this setting <I>must</I> be 1. Taking wxUSE_TOOLBAR_SIMPLE as an example:<P>If wxUSE_TOOLBAR_NATIVE is 0, wxUSE_TOOLBAR_SIMPLE must be 1.<BR>If wxUSE_TOOLBAR_NATIVE is 1, wxUSE_TOOLBAR_SIMPLE may be 0 or 1."),
220 wxVariant(wxT(""), wxT("enabled-if-not")),
221 wxT("configitems")));
222
223 m_properties.AddProperty(
224 new ctProperty(
225 wxT("<B>Exclusivity</B><P> The settings that are mutually exclusive with this one."),
226 wxVariant(wxT(""), wxT("exclusivity")),
227 wxT("configitems")));
228
229 m_properties.AddProperty(
230 new ctProperty(
231 wxT("<B>Context</B><P> A list of symbols (config settings), at least one of which must be enabled for this item to participate in dependency rules.<P>\nIf empty, this item will always be used in dependency rules.<P>\nMostly this will be used to specify the applicable platforms, but it can contain other symbols, for example compilers."),
232 wxVariant(wxT(""), wxT("context")),
233 wxT("configitems")));
234
235 m_properties.AddProperty(
236 new ctProperty(
237 wxT("<B>Configure-command</B><P> Configure command to generate if this is on."),
238 wxVariant(wxT(""), wxT("configure-command")),
239 wxT("multiline")));
240
241 m_properties.AddProperty(
242 new ctProperty(
243 wxT("<B>Help-topic</B><P> The help topic in the wxWindows manual for this component or setting."),
244 wxVariant(wxT(""), wxT("help-topic")),
245 wxT("multiline")));
246
247 m_properties.AddProperty(
248 new ctProperty(
249 wxT("<B>Notes</B><P> User notes."),
250 wxVariant(wxT(""), wxT("notes")),
251 wxT("multiline")));
252
253 m_defaultProperty = wxT("description");
254 }
255
256 /// Do additional actions to apply the property to the internal
257 /// representation.
258 void ctConfigItem::ApplyProperty(ctProperty* prop, const wxVariant& oldValue)
259 {
260 ctConfigToolDoc* doc = GetDocument();
261 bool oldModified = doc->IsModified();
262 doc->Modify(TRUE);
263
264 wxString name = prop->GetName();
265 if (name == wxT("requires") ||
266 name == wxT("precludes") ||
267 name == wxT("enabled-if") ||
268 name == wxT("enabled-if-not") ||
269 name == wxT("context"))
270 {
271 doc->RefreshDependencies();
272 }
273 if (doc && doc->GetFirstView() && oldModified != doc->IsModified())
274 ((ctConfigToolView*)doc->GetFirstView())->OnChangeFilename();
275 }
276
277 /// Get the associated document (currently, assumes
278 /// there's only ever one document active)
279 ctConfigToolDoc* ctConfigItem::GetDocument()
280 {
281 ctConfigToolDoc* doc = wxGetApp().GetMainFrame()->GetDocument();
282 return doc;
283 }
284
285 /// Convert string containing config item names to
286 /// an array of config item names
287 void ctConfigItem::StringToArray(const wxString& items, wxArrayString& itemsArray)
288 {
289 wxStringTokenizer tokenizer(items, wxT(","));
290 while (tokenizer.HasMoreTokens())
291 {
292 wxString token = tokenizer.GetNextToken();
293 itemsArray.Add(token);
294 }
295 }
296
297 /// Convert array containing config item names to
298 /// a string
299 void ctConfigItem::ArrayToString(const wxArrayString& itemsArray, wxString& items)
300 {
301 items = wxEmptyString;
302 size_t i;
303 for (i = 0; i < itemsArray.GetCount(); i++)
304 {
305 items += itemsArray[i];
306 if (i < (itemsArray.GetCount() - 1))
307 items += wxT(",");
308 }
309 }
310
311 /// Populate a list of items found in the string.
312 void ctConfigItem::StringToItems(ctConfigItem* topItem, const wxString& items, wxList& list)
313 {
314 wxArrayString strArray;
315 StringToArray(items, strArray);
316 size_t i;
317 for (i = 0; i < strArray.GetCount(); i++)
318 {
319 wxString str(strArray[i]);
320 ctConfigItem* item = topItem->FindItem(str);
321 if (item)
322 list.Append(item);
323 }
324 }
325
326 /// Find an item in this hierarchy
327 ctConfigItem* ctConfigItem::FindItem(const wxString& name)
328 {
329 if (GetName() == name)
330 return this;
331
332 for ( wxNode* node = GetChildren().GetFirst(); node; node = node->GetNext() )
333 {
334 ctConfigItem* child = (ctConfigItem*) node->GetData();
335 ctConfigItem* found = child->FindItem(name);
336 if (found)
337 return found;
338 }
339 return NULL;
340 }
341
342 /// Find the next sibling
343 ctConfigItem* ctConfigItem::FindNextSibling()
344 {
345 if (!GetParent())
346 return NULL;
347 wxNode* node = GetParent()->GetChildren().Member(this);
348 if (node && node->GetNext())
349 {
350 return (ctConfigItem*) node->GetNext()->GetData();
351 }
352 return NULL;
353 }
354
355 /// Find the previous sibling
356 ctConfigItem* ctConfigItem::FindPreviousSibling()
357 {
358 if (!GetParent())
359 return NULL;
360 wxNode* node = GetParent()->GetChildren().Member(this);
361 if (node && node->GetPrevious())
362 {
363 return (ctConfigItem*) node->GetPrevious()->GetData();
364 }
365 return NULL;
366 }
367
368 /// Sync appearance
369 void ctConfigItem::Sync()
370 {
371 if (GetDocument())
372 {
373 ctConfigToolView* view = (ctConfigToolView*) GetDocument()->GetFirstView();
374 if (view)
375 {
376 view->SyncItem(wxGetApp().GetMainFrame()->GetConfigTreeCtrl(), this);
377 }
378 }
379 }
380
381 /// Create a clone of this and children
382 ctConfigItem* ctConfigItem::DeepClone()
383 {
384 ctConfigItem* newItem = Clone();
385
386 for ( wxNode* node = GetChildren().GetFirst(); node; node = node->GetNext() )
387 {
388 ctConfigItem* child = (ctConfigItem*) node->GetData();
389 ctConfigItem* newChild = child->DeepClone();
390 newItem->AddChild(newChild);
391 }
392 return newItem;
393 }
394
395 /// Detach: remove from parent, and remove tree items
396 void ctConfigItem::Detach()
397 {
398 if (GetParent())
399 GetParent()->RemoveChild(this);
400 else
401 GetDocument()->SetTopItem(NULL);
402 SetParent(NULL);
403
404 wxTreeItemId treeItem = GetTreeItemId();
405
406 DetachFromTree();
407
408 // Will delete the branch, but not the config items.
409 wxGetApp().GetMainFrame()->GetConfigTreeCtrl()->Delete(treeItem);
410 }
411
412 /// Hide from tree: make sure tree deletions won't delete
413 /// the config items
414 void ctConfigItem::DetachFromTree()
415 {
416 wxTreeItemId item = GetTreeItemId();
417
418 ctTreeItemData* data = (ctTreeItemData*) wxGetApp().GetMainFrame()->GetConfigTreeCtrl()->GetItemData(item);
419 data->SetConfigItem(NULL);
420 m_treeItemId = wxTreeItemId();
421
422 for ( wxNode* node = GetChildren().GetFirst(); node; node = node->GetNext() )
423 {
424 ctConfigItem* child = (ctConfigItem*) node->GetData();
425 child->DetachFromTree();
426 }
427 }
428
429 /// Attach: insert after the given position
430 void ctConfigItem::Attach(ctConfigItem* parent, ctConfigItem* insertBefore)
431 {
432 if (parent)
433 {
434 SetParent(parent);
435 wxNode* node = NULL;
436 if (insertBefore)
437 node = parent->GetChildren().Member(insertBefore);
438
439 if (node)
440 parent->GetChildren().Insert(node, this);
441 else
442 parent->GetChildren().Append(this);
443 }
444 else
445 {
446 GetDocument()->SetTopItem(this);
447 }
448 }
449
450 /// Can have children?
451 bool ctConfigItem::CanHaveChildren() const
452 {
453 return (GetType() == ctTypeGroup ||
454 GetType() == ctTypeCheckGroup ||
455 GetType() == ctTypeRadioGroup);
456 }
457
458 // An item is in the active context if:
459 // The context field is empty; or
460 // The context field contains a symbol that is currently enabled.
461 bool ctConfigItem::IsInActiveContext()
462 {
463 wxString context = GetPropertyString(wxT("context"));
464 if (context.IsEmpty())
465 return TRUE;
466
467 wxList contextItems;
468 StringToItems(GetDocument()->GetTopItem(), context, contextItems);
469
470 for ( wxNode* node = contextItems.GetFirst(); node; node = node->GetNext() )
471 {
472 ctConfigItem* otherItem = (ctConfigItem*) node->GetData();
473 if (otherItem->IsEnabled())
474 return TRUE;
475 }
476 return FALSE;
477 }
478
479 /// Evaluate the requires properties:
480 /// if any of the 'requires' items are disabled,
481 /// then this one is disabled (and inactive).
482 void ctConfigItem::EvaluateDependencies()
483 {
484 wxList items;
485 wxString requires = GetPropertyString(wxT("requires"));
486 wxString precludes = GetPropertyString(wxT("precludes"));
487 wxString enabledIf = GetPropertyString(wxT("enabled-if"));
488 wxString enabledIfNot = GetPropertyString(wxT("enabled-if-not"));
489
490 bool active = TRUE;
491 bool enabled = IsEnabled();
492 bool oldEnabled = enabled;
493 bool oldActive = IsActive();
494 bool explicitlyEnabled = FALSE;
495 bool explicitlyDisabled = FALSE;
496 bool inActiveContext = IsInActiveContext();
497
498 // Add the parent to the list of dependencies, if the
499 // parent is a check or radio group.
500 ctConfigItem* parent = GetParent();
501 if (parent &&
502 (parent->GetType() == ctTypeCheckGroup ||
503 parent->GetType() == ctTypeRadioGroup))
504 items.Append(parent);
505
506 wxList tempItems;
507 StringToItems(GetDocument()->GetTopItem(), requires, tempItems);
508
509 wxNode* node;
510 for ( node = tempItems.GetFirst(); node; node = node->GetNext() )
511 {
512 // Only consider the dependency if both items are in
513 // an active context.
514 // Each is in the active context if:
515 // The context field is empty; or
516 // The context field contains a symbol that is currently enabled.
517 ctConfigItem* otherItem = (ctConfigItem*) node->GetData();
518 if (inActiveContext && otherItem->IsInActiveContext())
519 items.Append(otherItem);
520 }
521
522 {
523 int enabledCount = 0;
524 for ( node = items.GetFirst(); node; node = node->GetNext() )
525 {
526 ctConfigItem* otherItem = (ctConfigItem*) node->GetData();
527
528 if (otherItem->IsEnabled())
529 {
530 enabledCount ++;
531 }
532 }
533 if (items.GetCount() > 0 && enabledCount == 0)
534 {
535 // None of the items were enabled
536 enabled = FALSE;
537 active = FALSE;
538 explicitlyDisabled = TRUE;
539 }
540 }
541
542 items.Clear();
543 if (!enabledIfNot.IsEmpty())
544 {
545 StringToItems(GetDocument()->GetTopItem(), enabledIfNot, items);
546 int disabledCount = 0;
547 int inContextCount = 0;
548
549 for ( wxNode* node = items.GetFirst(); node; node = node->GetNext() )
550 {
551 ctConfigItem* otherItem = (ctConfigItem*) node->GetData();
552
553 if (inActiveContext && otherItem->IsInActiveContext())
554 {
555 // Make this enabled and inactive, _unless_ it's
556 // already been explicitly disabled in the previous
557 // requires evaluation (it really _has_ to be off)
558 if (!otherItem->IsEnabled())
559 {
560 disabledCount ++;
561 }
562 inContextCount ++;
563 }
564 }
565 // Enable if there were no related items that were enabled
566 if (inContextCount > 0 && (disabledCount == inContextCount) && !explicitlyDisabled)
567 {
568 explicitlyEnabled = TRUE;
569 enabled = TRUE;
570 active = FALSE;
571 }
572 }
573
574 items.Clear();
575 if (!enabledIf.IsEmpty())
576 {
577 StringToItems(GetDocument()->GetTopItem(), enabledIf, items);
578 int enabledCount = 0;
579 int inContextCount = 0;
580
581 for ( wxNode* node = items.GetFirst(); node; node = node->GetNext() )
582 {
583 ctConfigItem* otherItem = (ctConfigItem*) node->GetData();
584
585 if (inActiveContext && otherItem->IsInActiveContext())
586 {
587 // Make this enabled and inactive, _unless_ it's
588 // already been explicitly disabled in the previous
589 // requires evaluation (it really _has_ to be off)
590 if (otherItem->IsEnabled())
591 {
592 enabledCount ++;
593 }
594 inContextCount ++;
595 }
596 }
597 // Enable if there were no related items that were disabled
598 if (inContextCount > 0 && (enabledCount > 0) && !explicitlyDisabled)
599 {
600 explicitlyEnabled = TRUE;
601 enabled = TRUE;
602 active = FALSE;
603 }
604 }
605
606 items.Clear();
607 if (!precludes.IsEmpty())
608 {
609 StringToItems(GetDocument()->GetTopItem(), precludes, items);
610 int enabledCount = 0;
611 int inContextCount = 0;
612
613 for ( wxNode* node = items.GetFirst(); node; node = node->GetNext() )
614 {
615 ctConfigItem* otherItem = (ctConfigItem*) node->GetData();
616
617 if (inActiveContext && otherItem->IsInActiveContext())
618 {
619 // Make this enabled and inactive, _unless_ it's
620 // already been explicitly disabled in the previous
621 // requires evaluation (it really _has_ to be off)
622 if (otherItem->IsEnabled())
623 {
624 enabledCount ++;
625 }
626 }
627 }
628 // Disable if there were no related items that were disabled
629 if (inContextCount > 0 && (enabledCount == inContextCount) && !explicitlyEnabled)
630 {
631 enabled = FALSE;
632 active = FALSE;
633 }
634 }
635
636 // Finally check a sort of dependency: whether our
637 // context is active. If not, make this inactive.
638 if (!IsInActiveContext())
639 active = FALSE;
640
641 SetActive(active);
642
643 // If going active, set enabled state to the default state
644 if (active &&
645 oldActive != active &&
646 (GetType() == ctTypeBoolCheck || GetType() == ctTypeCheckGroup) &&
647 m_properties.FindProperty(wxT("default-state")))
648 {
649 bool defaultState = m_properties.FindProperty(wxT("default-state"))->GetVariant().GetBool();
650 enabled = defaultState;
651 }
652 Enable(enabled);
653
654 // Deal with setting a radio button
655 if (enabled && enabled != oldEnabled &&
656 (GetType() == ctTypeBoolRadio || GetType() == ctTypeRadioGroup))
657 {
658 wxList considered;
659 PropagateRadioButton(considered);
660 }
661 }
662
663 /// Get description, which may be dynamically
664 /// generated depending on the property.
665 wxString ctConfigItem::GetDescription(ctProperty* property)
666 {
667 if (property->GetName() == wxT("description"))
668 {
669 wxString value(property->GetValue());
670 if (value.IsEmpty())
671 return wxT("Double-click on <B>description</B> to write a brief explanation of the setting.<P>");
672 else
673 return value;
674 }
675 else if (property->GetName() == wxT("notes"))
676 {
677 wxString value(property->GetValue());
678 if (value.IsEmpty())
679 return wxT("Double-click on <B>notes</B> to write notes about this setting.<P>");
680 else
681 return value;
682 }
683 return property->GetDescription();
684 }
685
686 /// Get the title for the property editor
687 wxString ctConfigItem::GetTitle()
688 {
689 wxString title(GetName());
690 if (GetType() == ctTypeCheckGroup ||
691 GetType() == ctTypeRadioGroup ||
692 GetType() == ctTypeBoolCheck ||
693 GetType() == ctTypeBoolRadio)
694 {
695 if (IsEnabled())
696 title = title + _T(" - enabled");
697 else
698 title = title + _T(" - disabled");
699 }
700 return title;
701 }
702
703 /// Propagate a change in enabled/disabled status
704 void ctConfigItem::PropagateChange(wxList& considered)
705 {
706 if (GetType() == ctTypeCheckGroup ||
707 GetType() == ctTypeRadioGroup ||
708 GetType() == ctTypeBoolCheck ||
709 GetType() == ctTypeBoolRadio)
710 {
711 // TODO: what about string, integer? Can they have
712 // dependencies?
713
714 for ( wxNode* node = GetDependents().GetFirst(); node; node = node->GetNext() )
715 {
716 ctConfigItem* child = (ctConfigItem*) node->GetData();
717
718 // Avoid loops
719 if (!considered.Member(child))
720 {
721 considered.Append(child);
722
723 child->EvaluateDependencies();
724 child->Sync();
725
726 child->PropagateChange(considered);
727 }
728 }
729 }
730 }
731
732 /// Process radio button selection
733 void ctConfigItem::PropagateRadioButton(wxList& considered)
734 {
735 if ((GetType() == ctTypeBoolRadio || GetType() == ctTypeRadioGroup) && IsEnabled())
736 {
737 wxString mutuallyExclusive(GetPropertyString(wxT("exclusivity")));
738
739 wxList list;
740 StringToItems(GetDocument()->GetTopItem(), mutuallyExclusive, list);
741
742 for ( wxNode* node = list.GetFirst(); node; node = node->GetNext() )
743 {
744 ctConfigItem* child = (ctConfigItem*) node->GetData();
745 if (child->IsEnabled() && child != this)
746 {
747 child->Enable(FALSE);
748 child->Sync();
749
750 if (!considered.Member(child))
751 child->PropagateChange(considered);
752 }
753 }
754 }
755 }