c# - How to cache dynamically switched views by ViewModel-First approach using DataTemplates -


(i using galasoft.mvvmlight framework)

i have views in mainwindow.xaml switching between them dynamically in run-time, user selection.

these views using following technique bind corresponding view-model:

mainwindow.xaml

... <window.resources>         <datatemplate datatype="{x:type vm:control1viewmodel}">             <v:control1/>         </datatemplate>     ... // assume there more 1 datatemplate. every view has unique view-model. </window.resources> ... 

control1viewmodel.cs

public class control1viewmodel : viewmodelbase {     ... } 

the mainwindow.xaml using following technique switch between above views:

mainwindow.xaml

... <contentcontrol content="{binding currentview}"/> // view appears. ... 

mainviewmodel.cs

public class mainviewmodel : viewmodelbase {     ...     private viewmodelbase _currentview;     public viewmodelbase currentview     {         { return _currentview; }         private set         {             _currentview = value;             base.raisepropertychanged("currentview");         }     }     ... } 

for convenience, didn't add more controls, put 1 (control1) shorten question code section. mentioned above, assume there more 1 view switch.

everytime currentview property set new viewmodelbase value (e.g. control1viewmodel), wpf construct new instance of binded view datatemplate visual tree object, old 1 lost.

that means cannot cache views (e.g. control1) while switching between them.

the solution had found answer "hardcode" view viewmodel (using datacontext), following solution bellow happens:

  • i breaking viewmodel-first approach.
  • in order not break complete mvvm, have change currentview signature , move code behind of mainwindow.xaml.
  • instead of switching viewmodelbase types, switching concrete controls.

i know if there solution without "hardcoding" view view-model, can keep current viewmodelbase switch , viewmodel-first approach.

you can have following approach:

  1. instead of taking contentcontrol take itemscontrol

<itemscontrol itemssource="{binding views}" selecteditem="{binding currentview}"/>

  1. take itemspanel of itemscontrol grid , selecteditem set z-index 1 , rest of items set z-index 0. in way 1 view visible @ time on other views.

  2. take 2 properties in mainviewmodel. views of type observablecollection<viewmodelbase> , currentview of type viewmodelbase , bind itemscontrols's itemssource , selecteditem respectively.

now when want open view, create viewmodel, add views list , set currentview. if there in list, set ascurrentview.

also provide close button if want closed forever. i.e. if close it, removed list , won't cached.

this work if have different views opened in window , can switch between them. , if want can close view.

edit: see below code:

            <itemscontrol itemssource="{binding views}">                 <itemscontrol.itemspanel>                     <itemspaneltemplate>                         <grid margin="10,10,0,10">                         </grid>                     </itemspaneltemplate>                 </itemscontrol.itemspanel>                  <itemscontrol.itemcontainerstyle>                     <style>                         <setter property="grid.opacity" value="{binding zindex}"/>                         <setter property="grid.zindex" value="{binding zindex}"/>                     </style>                 </itemscontrol.itemcontainerstyle>                 <itemscontrol.itemtemplate>                     <datatemplate>                         <contentcontrol content="{binding}"/>                     </datatemplate>                 </itemscontrol.itemtemplate>             </itemscontrol> 

here can see in viewmodel have have property zindex, used show current view on top. whenever want show view, set zindex property of viewmodel 1 , reset reset view 0.


Comments

Popular posts from this blog

apache - PHP Soap issue while content length is larger -

asynchronous - Python asyncio task got bad yield -

javascript - Complete OpenIDConnect auth when requesting via Ajax -