In my program I have tabItems
that have their commands bound to a View Model. I am in the process of implementing a function that will copy the design structure of a "master" tabItem
, along with it's command functionality in order to create a new tabItem
. I need to do this because the user of this program will be allowed to add new tabItems
.
Currently I am using the question Copying a TabItem with an MVVM structure, but I seem to be having trouble when the function tries to copy the Grid
object using dependencyValue
.
The class I am using:
public static class copyTabItems
{
public static IList<DependencyProperty> GetAllProperties(DependencyObject obj)
{
return (from PropertyDescriptor pd in TypeDescriptor.GetProperties(obj, new Attribute[] { new PropertyFilterAttribute(PropertyFilterOptions.SetValues) })
select DependencyPropertyDescriptor.FromProperty(pd)
into dpd
where dpd != null
select dpd.DependencyProperty).ToList();
}
public static void CopyPropertiesFrom(this FrameworkElement controlToSet,
FrameworkElement controlToCopy)
{
foreach (var dependencyValue in GetAllProperties(controlToCopy)
.Where((item) => !item.ReadOnly)
.ToDictionary(dependencyProperty => dependencyProperty, controlToCopy.GetValue))
{
controlToSet.SetValue(dependencyValue.Key, dependencyValue.Value);
}
}
}
When dependencyValue
gets to {[Content, System.Windows.Controls.Grid]}
the program throws an InvalidOperationException was Unhandled
stating that, "Specified element is already the logical child of another element. Disconnect it first".
What does this mean? Is this a common problem with the Grid
in WPF (am I breaking some rule by trying to do this?)? Is there something in my program that I am not aware of that is causing this?
Ok. This is how you're supposed to deal with a
TabControl
in WPF:Code Behind:
Main ViewModel:
Abstract TabItem ViewModel (you to derive from this to create each different Tab "Widget")
TabItem 1 ViewModel:
Edit: I forgot to post the Command class (though you surely have your own)
And finally PropertyChangedBase (just a helper class)
Result:
Widget
, which contains its own logic and Data.ViewModel
, and add it to theObservableCollection
. WPF's DataBinding automatically updates the UI based on the Collection's change notification.MVVMTabItemViewModel
and add your logic and data there. Then, you create aDataTemplate
for that new ViewModel and WPF takes care of the rest.bool
property in the ViewModel to which the UI binds.VisualTreeHelper.ComplicateMyCode()
kind of things.File -> New Project -> WPF Application
and see the results for yourself.