Click here to Skip to main content
6,592,897 members and growing! (17,640 online)
Email Password   helpLost your password?
Platforms, Frameworks & Libraries » Windows Presentation Foundation » Controls     Intermediate

WPF Control State Persistency

By Tomer Shamam

This article describes how to store and restore WPF's elements state, such as position, size, etc.
XML, C# 3.0WinXP, Win2003, Vista, .NET 3.0, XAML, WPF, VS2005, Dev
Posted:9 Jun 2007
Views:35,106
Bookmarked:29 times
Announcements
Loading...
 
Search    
Advanced Search
Add to IE Search
printPrint   add Share
      Discuss Discuss   Broken Article?Report  
10 votes for this article.
Popularity: 4.42 Rating: 4.42 out of 5

1

2
1 vote, 10.0%
3
3 votes, 30.0%
4
6 votes, 60.0%
5

Introduction

Someone asked me if WPF provides an option for storing or serializing control state. For an instance, having a ListView, is it possible to store the width of its columns after closing and opening the Window, or maybe after restarting the application?

I was thinking to myself, sure, you should use Data Binding. All you have to do is to bind the width or height, of any element to a back storage. For example you can create a State class for storing the data, and then you should bind it to your properties, using the Binding markup extension.

Thinking twice, I realized that it is much more complicated than it looks. Data Binding is a great tool but it should be customized to support this feature.

So the answer was no! but...

Then I developed a smart Markup Extension, backed up with Attached Properties and a smart Back Storage to provide an easy way to save controls properties state.

Using the code

To work with the attached code, you should:

  • Add a reference to the Tomers.WPF.State.dll assembly (unless you have compiled it with a different name)
  • Mark each persist-element with a unique ID (among the whole application) by using the ElementState.UId attached property
  • Select the state mode by using the ElementState.Mode attached property
  • Provide a value to any element property by using the ElementState Markup Extension

NOTE: I'm using the XmlnsDefinitionAttribute attribute for mapping the CLR namespace into the WPF XML root namespace, so you don't need to use any prefix for using these custom types from XAML.

The markup snippet below demonstrates how to use my Markup Extension and Attached Properties to store a Window Size and Position, and a ListView, GridViewColumn's width.

<Window x:Class="Tomers.WPF.State.Demo.DemoWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:data="clr-namespace:Tomers.WPF.State.Demo.Data"

    ElementState.Mode="Persist"
    ElementState.UId="DemoWindow"
    Height="{ElementState Default=300}"
    Width="{ElementState Default=300}"
    Left="{ElementState Default=100}"
    Top="{ElementState Default=100}"

    Title="Test">

    <Window.Resources>
        <ObjectDataProvider x:Key="cConsultants" 
            ObjectType="{x:Type data:Consultants}" />
    </Window.Resources>

    <StackPanel>
      <ListView ItemsSource="{Binding Source={StaticResource cConsultants}}">
        <ListView.View>
            <GridView ColumnHeaderToolTip="Sela Consultants">
               <GridViewColumn
                  ElementState.Mode="Persist"
                  ElementState.UId="DemoWindow_GridViewColumn1"
                  DisplayMemberBinding="{Binding Path=FirstName}"
                  Header="First Name"
                  Width="{ElementState Default=100}" />
               <GridViewColumn
                  ElementState.Mode="Persist"
                  ElementState.UId="DemoWindow_GridViewColumn2"
                  DisplayMemberBinding="{Binding Path=LastName}"
                  Header="Last Name"
                  Width="{ElementState Default=100}" />
               <GridViewColumn
                  ElementState.Mode="Persist"
                  ElementState.UId="DemoWindow_GridViewColumn3"
                  DisplayMemberBinding="{Binding Path=Blog}"
                  Header="Blog"
                  Width="{ElementState Default=Auto}" />
            </GridView>
        </ListView.View>
      </ListView>
    </StackPanel>
</Window>

Using the Demo

Download the attached file, extract it into your local directory, and open the WPFElementState.sln solution file with Visual Studio 2005. Build and Run!

Now, resize the main window, move it to any place on the desktop, resize any grid column. Close the application, run it again! (The Window position and size, and columns width should be restored).

How it Works?

As you can see, I'm using the ElementState.Mode and ElementState.UId attached properties to tell the back storage to save the state for these element's dependency properties. Then I'm using the ElementState Markup Extension to set each property and its default value.


The ElementState.Mode attached property can be one of: Persist or Memory values.

  • Persist is used to serialize the element state into an XML stream. Restarting the application will restore this state.
  • Memory is used to hold the state only in memory. Restarting the application will not restore this state.

The ElementState.UId attached property is used to uniquely identify the element (this must be a unique name among all elements of the application).

To load and save state, you should call the ElementStateOperations.Load and ElementStateOperations.Save methods respectively. These methods accept any valid Stream instance, which should be readable or writeable stream respectively. This stream is used for serializing the state by using the XmlSerializer class. For example:

public partial class App : System.Windows.Application
{
    protected override void OnStartup(StartupEventArgs e)
    {
        using (Stream stream = File.Open("ElementStateDemo.xml",
           FileMode.OpenOrCreate, FileAccess.Read, FileShare.Read))
        {
            ElementStateOperations.Load(stream);
        }
        base.OnStartup(e);
    }

    protected override void OnExit(ExitEventArgs e)
    {
        using (Stream stream = File.Open("ElementStateDemo.xml",
           FileMode.Create, FileAccess.Write, FileShare.None))
        {
            ElementStateOperations.Save(stream);
        }
        base.OnExit(e);
    }
}

Points of Interest

It is a first prototype, it took me several hours to write it down, so use it at your own risk.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

Tomer Shamam


Member
Tomer Shamam is a senior consultant and a lecturer at Sela Group Company. Currently leads the Smart Client Team.

You can read more of Tomer's thoughts in his blog at:
http://blogs.microsoft.co.il/blogs/tomershamam

You can find more details about Sela Group at:
http://www.sela.co.il
Occupation: Team Leader
Company: Sela Group
Location: Israel Israel

Other popular Windows Presentation Foundation articles:

Article Top
You must Sign In to use this message board.
FAQ FAQ 
 
Noise Tolerance  Layout  Per page   
 Msgs 1 to 25 of 29 (Total in Forum: 29) (Refresh)FirstPrevNext
GeneralHandling Bound columns? Pinmembercolkaih5:58 26 Nov '08  
GeneralGlobalization problem Pinmemberyoavo2:31 23 Sep '08  
GeneralThe demo is never runing with vs2008? Pinmemberwangfeng00718:02 21 Sep '08  
GeneralRe: The demo is never runing with vs2008? PinmemberTomer Shamam21:56 21 Sep '08  
GeneralSave order of the columns Pinmemberyoavo22:18 26 Aug '08  
GeneralRe: Save order of the columns PinmemberTomer Shamam3:09 30 Aug '08  
QuestionRead XAML file PinmemberKrishnraj22:53 18 Jul '08  
AnswerRe: Read XAML file PinmemberTomer Shamam20:50 19 Jul '08  
GeneralRe: Read XAML file PinmemberKrishnraj20:05 20 Jul '08  
GeneralRe: Read XAML file PinmemberKrishnraj22:46 20 Jul '08  
GeneralRe: Read XAML file PinmemberTomer Shamam12:02 22 Jul '08  
QuestionNRE Setting Attached Property on Markup Extension in XAML PinmemberRobbyMon11:59 15 Jul '08  
AnswerRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberTomer Shamam21:41 15 Jul '08  
GeneralRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberRobbyMon4:12 16 Jul '08  
GeneralRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberTomer Shamam4:17 16 Jul '08  
GeneralRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberTomer Shamam4:29 16 Jul '08  
GeneralRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberRobbyMon5:17 16 Jul '08  
GeneralRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberTomer Shamam6:50 16 Jul '08  
GeneralRe: NRE Setting Attached Property on Markup Extension in XAML PinmemberRobbyMon11:50 16 Jul '08  
GeneralFixed version Pinmembersungwoo park16:04 19 Jun '08  
GeneralUpdate ProvideValue routine Pinmemberklange13:30 2 Jun '08  
QuestionVisual Studio 2008 RTM Designer Pinmemberyohoohoy9:03 22 Nov '07  
AnswerRe: Visual Studio 2008 RTM Designer PinmemberTomer Shamam10:06 22 Nov '07  
GeneralRe: Visual Studio 2008 RTM Designer Pinmemberyohoohoy16:54 22 Nov '07  
GeneralRe: Visual Studio 2008 RTM Designer PinmemberGonzalo Brusella11:58 28 Aug '08  

General General    News News    Question Question    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

PermaLink | Privacy | Terms of Use
Last Updated: 9 Jun 2007
Editor: Deeksha Shenoy
Copyright 2007 by Tomer Shamam
Everything else Copyright © CodeProject, 1999-2009
Web19 | Advertise on the Code Project