6

我首先尝试过这种方法,但出现错误“元素已经是另一个元素的子元素”

var objClone = new MyImageControl();
objClone = this;
((Canvas)this.Parent).Children.Add(objClone);

然后我检查了thisthis,但 XamlWriter 和 XamlReader 在 WinRT 中不可用。我曾尝试使用MemberwiseClone()但它抛出异常,“无法使用已与其底层 RCW 分离的 COM 对象。System.Runtime.InteropServices.InvalidComObjectException”。那么谁能告诉我如何将画布中现有的 UserControl 克隆到自身?

4

2 回答 2

3

我编写了一个UIElement复制元素的属性和子元素的扩展——请注意,它不会克隆设置事件。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Windows.UI.Xaml;
using System.Reflection;
using Windows.UI.Xaml.Controls;

namespace UIElementClone
{
    public static class UIElementExtensions
    {
        public static T DeepClone<T>(this T source) where T : UIElement
        {

            T result;

            // Get the type
            Type type = source.GetType();

            // Create an instance
            result = Activator.CreateInstance(type) as T;

            CopyProperties<T>(source, result, type);

            DeepCopyChildren<T>(source, result);

            return result;
        }

        private static void DeepCopyChildren<T>(T source, T result) where T : UIElement
        {
            // Deep copy children.
            Panel sourcePanel = source as Panel;
            if (sourcePanel != null)
            {
                Panel resultPanel = result as Panel;
                if (resultPanel != null)
                {
                    foreach (UIElement child in sourcePanel.Children)
                    {
                        // RECURSION!
                        UIElement childClone = DeepClone(child);
                        resultPanel.Children.Add(childClone);
                    }
                }
            }
        }

        private static void CopyProperties<T>(T source, T result, Type type) where T : UIElement
        {
            // Copy all properties.

            IEnumerable<PropertyInfo> properties = type.GetRuntimeProperties();

            foreach (var property in properties)
            {
                if (property.Name != "Name") // do not copy names or we cannot add the clone to the same parent as the original.
                {
                    if ((property.CanWrite) && (property.CanRead))
                    {
                        object sourceProperty = property.GetValue(source);

                        UIElement element = sourceProperty as UIElement;
                        if (element != null)
                        {
                            UIElement propertyClone = element.DeepClone();
                            property.SetValue(result, propertyClone);
                        }
                        else
                        {
                            try
                            {
                                property.SetValue(result, sourceProperty);
                            }
                            catch (Exception ex)
                            {
                                System.Diagnostics.Debug.WriteLine(ex);
                            }
                        }
                    }
                }
            }
        }        
    }
}

如果您发现它有用,请随意使用此代码。

于 2013-05-20T08:00:52.167 回答
1

您可以尝试使用 XamlWriter 和 XamlReader 以外的序列化程序来实现链接描述的相同效果。例如,使用 ServiceStack.Text 将您的对象 JSON 序列化为字符串,然后从该字符串中获取一个新对象并将其添加到父对象。

于 2013-03-22T07:30:14.260 回答