Xml 数据到 WPF TreeView 的双向绑定
Posted
技术标签:
【中文标题】Xml 数据到 WPF TreeView 的双向绑定【英文标题】:Two-way binding of Xml data to the WPF TreeView 【发布时间】:2010-09-16 07:46:53 【问题描述】:我正在尝试使用 WPF 作为表示层重写我的ForestPad 应用程序。在 WinForms 中,我以编程方式填充每个节点,但如果可能的话,我想利用 WPF 的数据绑定功能。
一般而言,将 WPF TreeView 双向数据绑定到 Xml 文档的最佳方法是什么?
通用解决方案很好,但作为参考,我尝试绑定的 Xml 文档的结构如下所示:
<?xml version="1.0" encoding="utf-8"?>
<forestPad
guid="6c9325de-dfbe-4878-9d91-1a9f1a7696b0"
created="5/14/2004 1:05:10 AM"
updated="5/14/2004 1:07:41 AM">
<forest
name="A forest node"
guid="b441a196-7468-47c8-a010-7ff83429a37b"
created="01/01/2003 1:00:00 AM"
updated="5/14/2004 1:06:15 AM">
<data>
<![CDATA[A forest node
This is the text of the forest node.]]>
</data>
<tree
name="A tree node"
guid="768eae66-e9df-4999-b950-01fa9be1a5cf"
created="5/14/2004 1:05:38 AM"
updated="5/14/2004 1:06:11 AM">
<data>
<![CDATA[A tree node
This is the text of the tree node.]]>
</data>
<branch
name="A branch node"
guid="be4b0993-d4e4-4249-8aa5-fa9c940ae2be"
created="5/14/2004 1:06:00 AM"
updated="5/14/2004 1:06:24 AM">
<data>
<![CDATA[A branch node
This is the text of the branch node.]]></data>
<leaf
name="A leaf node"
guid="9c76ff4e-3ae2-450e-b1d2-232b687214aa"
created="5/14/2004 1:06:26 AM"
updated="5/14/2004 1:06:38 AM">
<data>
<![CDATA[A leaf node
This is the text of the leaf node.]]>
</data>
</leaf>
</branch>
</tree>
</forest>
</forestPad>
【问题讨论】:
【参考方案1】:好吧,如果你的元素层次结构更像...
<node type="forest">
<node type="tree">
...
...而不是您当前的架构。
按原样,您需要 4 个HierarchicalDataTemplate
s,一个用于包括根在内的每个分层元素,一个DataTemplate
用于leaf
元素:
<Window.Resources>
<HierarchicalDataTemplate
DataType="forestPad"
ItemsSource="Binding XPath=forest">
<TextBlock
Text="a forestpad" />
</HierarchicalDataTemplate>
<HierarchicalDataTemplate
DataType="forest"
ItemsSource="Binding XPath=tree">
<TextBox
Text="Binding XPath=data" />
</HierarchicalDataTemplate>
<HierarchicalDataTemplate
DataType="tree"
ItemsSource="Binding XPath=branch">
<TextBox
Text="Binding XPath=data" />
</HierarchicalDataTemplate>
<HierarchicalDataTemplate
DataType="branch"
ItemsSource="Binding XPath=leaf">
<TextBox
Text="Binding XPath=data" />
</HierarchicalDataTemplate>
<DataTemplate
DataType="leaf">
<TextBox
Text="Binding XPath=data" />
</DataTemplate>
<XmlDataProvider
x:Key="dataxml"
XPath="forestPad" Source="D:\fp.xml">
</XmlDataProvider>
</Window.Resources>
您可以改为以编程方式设置XmlDataProvider
的Source
:
dp = this.FindResource( "dataxml" ) as XmlDataProvider;
dp.Source = new Uri( @"D:\fp.xml" );
此外,重新保存您的编辑就像这样简单:
dp.Document.Save( dp.Source.LocalPath );
TreeView
本身需要一个Name
和一个绑定到XmlDataProvider
的ItemsSource
:
<TreeView
Name="treeview"
ItemsSource="Binding Source=StaticResource dataxml, XPath=.">
在这个示例中,我在每个节点上使用 TextBox
es 进行了 TwoWay
绑定,但是当在单独的单个 TextBox
或其他控件中一次仅编辑一个节点时,您将绑定到TreeView
的当前选中项。您还可以将上面的TextBox
es 更改为TextBlock
s,因为单击TextBox
并不会实际选择相应的TreeViewItem
。
<TextBox
DataContext="Binding ElementName=treeview, Path=SelectedItem"
Text="Binding XPath=data, UpdateSourceTrigger=PropertyChanged"/>
你必须使用两个Binding
s的原因是你不能同时使用Path
和XPath
。
编辑:
Timothy Lee Russell 询问有关将 CDATA 保存到数据元素的问题。首先,介绍一下InnerXml
和InnerText
。
在幕后,XmlDataProvider
使用XmlDocument
,它的树是XmlNodes
。当将诸如“stuff”之类的字符串分配给XmlNode
的InnerXml
属性时,这些标签就是真正的标签。获取或设置InnerXml
时不进行转义,解析为XML。
但是,如果将其分配给InnerText
属性,则尖括号将使用实体 < 进行转义。和>。取回值时会发生相反的情况。实体(如 <)被解析回字符(如
因此,如果我们存储在数据元素中的字符串包含 XML,则实体已被转义,我们需要通过在添加 CDATA 部分作为节点的子节点之前检索InnerText
来撤消它...
XmlDocument doc = dp.Document;
XmlNodeList nodes = doc.SelectNodes( "//data" );
foreach ( XmlNode node in nodes )
string data = node.InnerText;
node.InnerText = "";
XmlCDataSection cdata = doc.CreateCDataSection( data );
node.AppendChild( cdata );
doc.Save( dp.Source.LocalPath );
如果节点已经有一个 CDATA 部分并且值没有以任何方式更改,那么它仍然有一个 CDATA 部分,我们基本上用相同的部分替换它。但是,通过我们的绑定,如果我们更改数据元素内容的值,它会替换 CDATA 以支持转义字符串。然后我们必须修复它们。
【讨论】:
谢谢乔尔,这行得通。一个问题。我用 CDATA 部分围绕数据元素中的内容,以便可以存储 Xml。有没有办法控制 XmlDataProvider 如何写出数据元素? 如果有 XML 作为字符串,它将用实体转义尖括号(它们以 & 开头)。这可以逆转,因为 Document 属性返回一个 XmlDocument。我将在数据元素中编辑并添加用于执行 CDATA 的代码。 太棒了——行得通。对于我正在处理的文档的大小,性能确实很差,但我不会更新每个节点,而是添加一个 IsDirty 标志,并且只更新已编辑的节点。 我也不想要根节点,所以我将 TreeView 的 ItemsSource 中的 XPath 更改为“XPath=forest”而不是“XPath=”。效果很好。 我意识到这是一个非常古老的问答,但是今天这个答案对我帮助很大,所以我想谢谢你。不过,我很好奇,如果 XML 是node
树,如您所建议的,解决方案会有什么不同?【参考方案2】:
我们遇到了类似的问题。您可能会发现阅读 this article 很有帮助。我们使用了所描述的 ViewModel 模式,它简化了一切。
【讨论】:
【参考方案3】:我知道这是一篇旧帖子,但有一个更优雅的解决方案。如果您使用 XPath 表达式来选择您希望模板使用的所有节点,您确实可以使用单个 HierarchicalDataTemplate
:XPath=tree|branch|leaf
。
<HierarchicalDataTemplate x:Key="forestTemplate"
ItemsSource="Binding XPath=tree|branch|leaf">
<TextBlock Text="Binding XPath=data" />
</HierarchicalDataTemplate>
这是一个完整的 Page
示例,其中 XData 嵌入在 XmlDataProvider1
中:
<Page
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<Page.Resources>
<XmlDataProvider x:Key="sampleForestPad" XPath="forestPad/forest">
<x:XData>
<forestPad xmlns=""
guid="6c9325de-dfbe-4878-9d91-1a9f1a7696b0"
created="5/14/2004 1:05:10 AM"
updated="5/14/2004 1:07:41 AM">
<forest
name="A forest node"
guid="b441a196-7468-47c8-a010-7ff83429a37b"
created="01/01/2003 1:00:00 AM"
updated="5/14/2004 1:06:15 AM">
<data>
<![CDATA[A forest node
This is the text of the forest node.]]>
</data>
<tree
name="A tree node"
guid="768eae66-e9df-4999-b950-01fa9be1a5cf"
created="5/14/2004 1:05:38 AM"
updated="5/14/2004 1:06:11 AM">
<data>
<![CDATA[A tree node
This is the text of the tree node.]]>
</data>
<branch
name="A branch node"
guid="be4b0993-d4e4-4249-8aa5-fa9c940ae2be"
created="5/14/2004 1:06:00 AM"
updated="5/14/2004 1:06:24 AM">
<data>
<![CDATA[A branch node
This is the text of the branch node.]]></data>
<leaf
name="A leaf node"
guid="9c76ff4e-3ae2-450e-b1d2-232b687214aa"
created="5/14/2004 1:06:26 AM"
updated="5/14/2004 1:06:38 AM">
<data>
<![CDATA[A leaf node
This is the text of the leaf node.]]>
</data>
</leaf>
</branch>
</tree>
</forest>
</forestPad>
</x:XData>
</XmlDataProvider>
<HierarchicalDataTemplate x:Key="forestTemplate"
ItemsSource="Binding XPath=tree|branch|leaf">
<TextBlock Text="Binding XPath=data" />
</HierarchicalDataTemplate>
<Style TargetType="TreeViewItem">
<Setter Property="IsExpanded" Value="True"/>
</Style>
</Page.Resources>
<TreeView ItemsSource="Binding Source=StaticResource sampleForestPad"
ItemTemplate="StaticResource forestTemplate"/>
</Page>
这将呈现为:
【讨论】:
以上是关于Xml 数据到 WPF TreeView 的双向绑定的主要内容,如果未能解决你的问题,请参考以下文章