Как вы можете клонировать объект WPF?
У кого-нибудь есть хороший пример того, как глубоко клонировать объект WPF, сохраняя привязки данных?
Отмеченный ответ - первая часть.
Вторая часть состоит в том, что вам нужно создать ExpressionConverter и ввести его в процесс сериализации. Подробности здесь:
http://www.codeproject.com/KB/WPF/xamlwriterandbinding.aspx?fid=1428301&df=90&mpp=25&noise=3&sort=Position&view=Quick&select=2801571
Ответы
Ответ 1
Самый простой способ, которым я это сделал, - использовать 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);
Ответ 2
В .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();
Ответ 3
Здесь есть отличные ответы. Очень полезно. Я попробовал различные подходы для копирования информации привязки, включая подход, описанный в 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.");
}
}
}
Ответ 4
Как насчет:
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;
}
}
Конечно, этот глубокий клонит любой объект, и это может быть не самое быстрое решение в городе, но оно имеет наименьшее обслуживание...:)