有人有一个很好的例子如何深度克隆 WPF 对象,保留数据绑定吗?


标记的答案是第一部分。

第二部分是您必须创建一个 ExpressionConverter 并将其注入序列化过程中。详细信息如下:
http://www.codeproject.com/KB/WPF/xamlwriterandbinding.aspx?fid=1428301&df=90&mpp=25&noise=3&sort=Position&view=Quick&select=2801571

有帮助吗?

解决方案

我完成此操作的最简单方法是使用 XamlWriter 将 WPF 对象保存为字符串。Save 方法将序列化逻辑树中的对象及其所有子对象。现在您可以创建一个新对象并使用 XamlReader 加载它。

前任:将对象写入 xaml(假设该对象是网格控件):

string gridXaml = XamlWriter.Save(myGrid);

将其加载到一个新对象中:

StringReader stringReader = new StringReader(gridXaml);
XmlReader xmlReader = XmlReader.Create(stringReader);
Grid newGrid = (Grid)XamlReader.Load(xmlReader);

其他提示

在 .NET 4.0 中,新的 xaml 序列化堆栈使这变得更加容易。

var sb = new StringBuilder();
var writer = XmlWriter.Create(sb, new XmlWriterSettings
{
    Indent = true,
    ConformanceLevel = ConformanceLevel.Fragment,
    OmitXmlDeclaration = true,
    NamespaceHandling = NamespaceHandling.OmitDuplicates, 
});
var mgr = new XamlDesignerSerializationManager(writer);

// HERE BE MAGIC!!!
mgr.XamlWriterMode = XamlWriterMode.Expression;
// THERE WERE MAGIC!!!

System.Windows.Markup.XamlWriter.Save(this, mgr);
return sb.ToString();

这里有一些很好的答案。很有帮助。我尝试了各种复制绑定信息的方法,包括中概述的方法 http://pjlcon.wordpress.com/2011/01/14/change-a-wpf-binding-from-sync-to-async-programatically/ 但这里的信息是网上最好的!

我创建了一个可重用的扩展方法来处理 InvalidOperationException“使用后无法更改绑定”。 在我的场景中,我正在维护一些有人编写的代码,在对 DevExpress DXGrid 框架进行重大升级后,它不再起作用。以下完美解决了我的问题。返回对象的代码部分可能会更好,稍后我将对其进行重构。

/// <summary>
/// Extension methods for the WPF Binding class.
/// </summary>
public static class BindingExtensions
{
    public static BindingBase CloneViaXamlSerialization(this BindingBase binding)
    {
        var sb = new StringBuilder();
        var writer = XmlWriter.Create(sb, new XmlWriterSettings
        {
            Indent = true,
            ConformanceLevel = ConformanceLevel.Fragment,
            OmitXmlDeclaration = true,
            NamespaceHandling = NamespaceHandling.OmitDuplicates,
        });
        var mgr = new XamlDesignerSerializationManager(writer);

        // HERE BE MAGIC!!!
        mgr.XamlWriterMode = XamlWriterMode.Expression;
        // THERE WERE MAGIC!!!

        System.Windows.Markup.XamlWriter.Save(binding, mgr);
        StringReader stringReader = new StringReader(sb.ToString());
        XmlReader xmlReader = XmlReader.Create(stringReader);
        object newBinding = (object)XamlReader.Load(xmlReader);
        if (newBinding == null)
        {
            throw new ArgumentNullException("Binding could not be cloned via Xaml Serialization Stack.");
        }

        if (newBinding is Binding)
        {
            return (Binding)newBinding;
        }
        else if (newBinding is MultiBinding)
        {
            return (MultiBinding)newBinding;
        }
        else if (newBinding is PriorityBinding)
        {
            return (PriorityBinding)newBinding;
        }
        else
        {
            throw new InvalidOperationException("Binding could not be cast.");
        }
    }
}

怎么样:

    public static T DeepClone<T>(T from)
    {
        using (MemoryStream s = new MemoryStream())
        {
            BinaryFormatter f = new BinaryFormatter();
            f.Serialize(s, from);
            s.Position = 0;
            object clone = f.Deserialize(s);

            return (T)clone;
        }
    }

当然,这个深度克隆任何对象,它可能不是城里最快的解决方案,但它的维护最少......:)

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top