修改:经过研究,发现只要在Frame上设置JournalEntry.KeepAlive="True"就可以使用第一种绑定到Source的最简单的办法来实现文中的效果。不用自己管理Page
这几天给给别人做了几个Demo,但觉得每个Demo都做一个工程太麻烦,不好管理,于是决定把每个Demo都各自用Page,然后通过一个列表,可以选择各个页面来查看。就如下图的效果:

当从左边选中一项后,右边的Frame会展示相应的内容。
我采用了一个xml文件来描述相关的信息,示例如下:
<?
xml version="1.0" encoding="utf-8"
?>
<
Pages
>
<
Page
Name
="Page1"
Uri
="/FrameContent;Component/Page1.xaml"
/>
<
Page
Name
="Page2"
Uri
="/FrameContent;Component/Page2.xaml"
/>
</
Pages
>
这是一个非常简单的描述,每个Page项描述这个Page的Name和对应的xaml文件,然后我们可以简单用下面的方式来绑定这些信息来达到目的:
<
ListBox
x:Name
="ListBox1"
ItemsSource
="
{Binding Source={StaticResource ListSource}}
"
DisplayMemberPath
="@Name"
/>
<!--
最简单的方式就是用Frame的Source来做绑定,不过这样会导致每次都生成新的实例
-->
<
Frame
DataContext
="
{Binding ElementName=ListBox1, Path=SelectedItem}
"
Source
="
{Binding XPath=@Uri}
"
/>
不过,这样有一个小小的问题在于,每次我切换Page的时候,都将重新生成一个Page的实例,以前在Page上做的一些操作会消失。这不是我想要的,我希望有一个类似于PagePool的东西来缓存我这些创建好的页面,这样在切换的时候会减少创建实例的开销,同时可以保证无论怎么切换始终是同一个Page的实例。于是我做了如下的更改:
<
ListBox
x:Name
="ListBox1"
ItemsSource
="
{Binding Source={StaticResource ListSource}}
"
DisplayMemberPath
="@Name"
/>
<
Frame
x:Name
="PageHost1"
NavigationUIVisibility
="Hidden"
Content
="
{Binding ElementName=ListBox1, Path=SelectedItem, Converter={StaticResource Converter}}
"
/>
我直接把Frame的Content属性绑定到选中的项上,并且用一个Converter来返回Page的实例,在Converter里面,我用一个Dictionary来保存创建过的页面,保证它们只会被创建一次:
public
class
FrameContentConverter : IValueConverter

{
// 使用Page池来减少创建Page的开销
private Dictionary<string, Page> _pagePool = new Dictionary<string, Page>();


IValueConverter Members#region IValueConverter Members

public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)

{
// 把Frame的Content绑定到ListBox的SelectedItem上,无法对变化做出响应
XmlElement node = value as XmlElement;
if (node != null)

{
string name = node.Attributes["Name"].Value;
string uri = node.Attributes["Uri"].Value;

if (this._pagePool.ContainsKey(name))

{
return this._pagePool[name];
}
else

{
Page page = Application.LoadComponent(new Uri(uri, UriKind.Relative)) as Page;
this._pagePool.Add(name, page);
return page;
}
}
return null;
}

public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)

{
XmlElement node = value as XmlElement;
if (node != null)

{
string name = node.Attributes["Name"].Value;
string uri = node.Attributes["Uri"].Value;

if (this._pagePool.ContainsKey(name))

{
return this._pagePool[name];
}
else

{
Page page = Application.LoadComponent(new Uri(uri, UriKind.Relative)) as Page;
this._pagePool.Add(name, page);
return page;
}
}
return null;
}

#endregion
}
理论上来说,这应该工作良好,然而,实际上却发生一点小小的意外:

Frame展示的内容只跟第一次选中的项有关,其后无论如何切换,Frame的内容均不受影响
难道是Frame的Content属性默认是以OneTime的模式来绑定的?于是我显示地指定Mode=OneWay,结果依然如此。
莫非Frame的Content属性只能指定一次?好吧,既然正着来不行,那我就试试反着的。
索性,我把ListBox的SeletedItem使用OneWayToSource的模式绑定到Frame的Content属性上。当然,这种情况下,FrameContentConverter里面的ConvertBack方法需要实现。
<
ListBox
x:Name
="ListBox2"
ItemsSource
="
{Binding Source={StaticResource ListSource}}
"
DisplayMemberPath
="@Name"
SelectedItem
="
{Binding ElementName=PageHost2, Path=Content, Converter={StaticResource Converter}, Mode=OneWayToSource}
"
/>
<
Frame
x:Name
="PageHost2"
NavigationUIVisibility
="Hidden"
/>
采用这种方式后,一切正常了:


这说明,Frame的Content属性不是只能设置一次的,只是在对Frame的Content属性做绑定时,由于某种未知的原因,无法对数据源的变化产生响应,鉴于我目前的系统是Vista SP1,上次的Frame出现了渲染上的问题,这次我也没有找没装SP1的机子做测试,所以我并不能确定这是WPF本身的BUG还是sp1引起的问题,还是我某个地方没有弄好造成的。
做了一个对比的示例,有兴趣的读者可以下载回去看看。http://files.cnblogs.com/RMay/FrameContent.rar
