Я новичок в WPF, пытаясь построить проект, который следует рекомендациям Джоша Смита, отличная статья, описывающая Шаблон проектирования Model-View-ViewModel.
Используя пример кода Josh в качестве базы, я создал простое приложение, которое содержит несколько "рабочих пространств", каждая из которых представлена вкладкой в TabControl. В моем приложении рабочая область - это редактор документов, который позволяет управлять иерархическим документом с помощью элемента управления TreeView.
Хотя мне удалось открыть несколько рабочих областей и просмотреть их содержимое в связанном элементе управления TreeView, я обнаружил, что TreeView "забывает" его состояние при переключении между вкладками. Например, если TreeView в Tab1 частично расширен, он будет показан как полностью свернутый после перехода на Tab2 и возврата к Tab1. Это поведение, по-видимому, относится ко всем аспектам состояния управления для всех элементов управления.
После некоторых экспериментов я понял, что могу сохранить состояние в TabItem, явно привязывая каждое свойство состояния управления к выделенному свойству в базовом ViewModel. Однако это кажется большой дополнительной работой, когда я просто хочу, чтобы все мои элементы управления запоминали их состояние при переключении между рабочими пространствами.
Я предполагаю, что мне не хватает чего-то простого, но я не уверен, где искать ответ. Любое руководство будет высоко оценено.
Спасибо, Тим
Update:
В соответствии с запросом я попытаюсь опубликовать код, демонстрирующий эту проблему. Однако, поскольку данные, лежащие в основе TreeView, сложны, я опубликую упрощенный пример, демонстрирующий те же самые симтомы. Вот XAML из главного окна:
<TabControl IsSynchronizedWithCurrentItem="True" ItemsSource="{Binding Path=Docs}">
<TabControl.ItemTemplate>
<DataTemplate>
<ContentPresenter Content="{Binding Path=Name}" />
</DataTemplate>
</TabControl.ItemTemplate>
<TabControl.ContentTemplate>
<DataTemplate>
<view:DocumentView />
</DataTemplate>
</TabControl.ContentTemplate>
</TabControl>
Вышеупомянутый XAML правильно привязывается к ObservableCollection DocumentViewModel, посредством чего каждый элемент представляется через DocumentView.
Для простоты этого примера я удалил TreeView (упомянутый выше) из DocumentView и заменил его TabControl, содержащим 3 фиксированных вкладки:
<TabControl>
<TabItem Header="A" />
<TabItem Header="B" />
<TabItem Header="C" />
</TabControl>
В этом случае между DocumentView и DocumentViewModel нет привязки. Когда код запускается, внутренний TabControl не может запомнить его выбор при переключении внешнего TabControl.
Однако, если я явно привязываю внутреннее свойство TabControl SelectedIndex...
<TabControl SelectedIndex="{Binding Path=SelectedDocumentIndex}">
<TabItem Header="A" />
<TabItem Header="B" />
<TabItem Header="C" />
</TabControl>
... к соответствующему фиктивному свойству в DocumentViewModel...
public int SelecteDocumentIndex { get; set; }
... внутренняя вкладка может запомнить ее выбор.
Я понимаю, что могу эффективно решить свою проблему, применив эту технику к каждому визуальному свойству каждого элемента управления, но я надеюсь, что есть более элегантное решение.