• 复制/粘贴数据库对象


    介绍 本文将遵循一个简单的场景,一个Restaurant对象类,它有几个属性,包括a List<比;HealthScore对象。 HealthScore对象将被复制和粘贴。就像示例中的类一样,它可以标记为Serializable,但这只是假设 在本文中,抽象概念对象将包含进一步的项,这些项排除了它是二进制可序列化的。因此,我们 还要实现一个姊妹对象ClipboardHealthScore。我们将来回转换这个姊妹对象,以便存储在剪贴板上。 我们还将看到如何劫持windows消息que来捕获剪贴板中的更改,从而允许我们动态地控制粘贴 按钮,基于剪贴板是否包含有效的可粘贴数据。 背景 为了将

      

    自定义对象存储在系统剪贴板中,它必须是二进制可序列化的。通常,在数据绑定对象中,可能会包含一些被排除的内容 它来自序列化资格,例如事件、专门的getter或属性setter以及其他此类性质的东西。因此,我们假设 我们要存储的对象不满足序列化要求。 使用的代码 首先,让我们花一点时间来讨论用于复制和粘贴函数的数据对象。我们感兴趣的产品是 HealthScore类型。 现在,就像我之前说的,在这种情况下,我们可以简单地使用我们的数据对象,因为它符合二进制序列化的条件,但我们假设它不符合, 作为现实生活中的对象,它很可能会更加复杂。出于这个原因,我们实现了一个姐妹类,以便于在剪贴板中进行存储。首先,我们的数据对象: 隐藏,复制Code

    public class HealthScore
    {
        private int _value;
        public HealthScore() { _value = 0; }
        public HealthScore(int value) { _value = value; }
        public int Value { get { return _value; } set { _value = value; } }
    }

    我们的姐妹类,ClipboardHealthScore将几乎相同,除了我们会扩大 IComareable< HealthScore>并包含一个int属性 SourceIndex。 这将允许我们在将集合存储到剪贴板之前对其进行排序。这很重要,因为当您单步执行选中单元格时 DataGridView, 您将按照它们被选中的顺序获得它们,而不是按照它们在表中出现的顺序。按索引排序将允许我们的粘贴在相同的顺序 与源表中的一样。延长IComparable<比;需要实施 CompareTo(ClipboardHealthScore a, ClipboardHealthScore b)方法,需要 根据是否<a>返回1、-1或0被认为大于<b>, 1表示是,-1表示否,如果它们相等,则为零。 另外,我们需要将这个类标记为Serializable: 隐藏,复制Code

    [Serializable]
    public class ClipboardHealthScore : IComparable<ClipboardHealthScore>;
    {
        private int _value;
        private int _sourceindex;
        public ClipboardHealthScore() { _value = _sourceindex= 0; }
        public ClipboardHealthScore(int value, int sourceindex) { _value = value; _sourceindex = sourceindex; }
        public int Value { get { return _value; } set { _value = value; } }
        public int SourceIndex { get { return _sourceindex; } set { _sourceindex = value; } }
        public int CompareTo(ClipboardHealthScore a, ClipboardHealthScore b)
        {
            if (a.SourceIndex > b.SourceIndex) return 1;    
            else if (a.SourceIndex < b.SourceIndex) return -1;
            else return 0;
        }
    }

    另外,一个正常的列表。集合是不可序列化的,这只是因为它的基类没有这样标记。因此,我们需要创建一个类 扩展List< T>要将我们的ClipboardHealthScore条目存储在其中,并将其标记为Serializable: 隐藏,复制Code

    [Serializable]
    public class ClipboardHealthScoreCollection : List<ClipboardHealthScore>
    {
        public ClipboardHealthScoreCollection() { }
    }

    现在复制,我们将获取选定的健康分数,将它们转换为 并将它们存储在剪贴板上。要粘贴,我们将检索 clipboardhealthscore从剪贴板返回并将其转换为正常 并将其添加到我们的数据对象中。 首先,让我们复制我们选择的项目: 隐藏,复制Code

    private void copybutton_Click(object sender, EventArgs e)
    {
        // First lets store each selected cell's row index in a list
        // making sure not to store duplicates
        List<Int32> selectedindexes = new List<Int32>();
        foreach (DataGridViewCell cell in this.dataGridView1.SelectedCells)
        { 
            if (!selectedindexes.Contains(c.RowIndex)) selectedindexes.Add(cell.RowIndex);
        }
        // Now we step through these indices converting each item to the ClipboardHealthScore
        // and  putting it into a collection, then add the collection to the Clipboard
        ClipboardHealthScoreCollection copyitems = new ClipboardHealthScoreCollection();
        foreach(int i in selectedindexes)
        {
            copyitems.Add(new ClipboardHealthScore(this.healthScoreBindingSource[i].Value, i));
        } 
        copyitems.Sort();
        DataObject dobj = new DataObject();
        dobj.SetData(typeof(ClipboardHealthScoreCollection), copyitems);
        Clipboard.SetDataObject(dobj);
    }

    不是太难。现在让我们处理粘贴这些项目到我们的数据集: 隐藏,复制Code

    private void pastebutton_Click(object sender, EventArgs e)
    {
        DataObject dobj = (DataObject)Clipboard.GetDataObject();
    
        if (dobj.GetDataPresent(typeof(ClipboardHealthScoreCollection)))
        {
            ClipboardHealthScoreCollection pastescores = 
              (dobj.GetData(typeof(ClipboardHealthScoreCollection))as ClipboardHealthscoreCollection);
      
            HealthScore newscore;
            foreach(ClipboardHealthScore hs in pastescores)
            {
                newscore = new HealthScore();
                newscore.Value = hs.Value;
                AddHealthScore(newscore);
            }
        }
    }

    最后,让我们讨论一下如何设置您的表单来侦听剪贴板的变化,从而允许您相应地处理粘贴控件。如果您正在制作一个MDI应用程序, 我建议您将MDI父窗体设置为侦听器。 首先,我们需要定义两个常量int变量来定义我们想要监听的Windows消息。 隐藏,复制Code

    private const int WM_DRAWCLIPBOARD = 0x0308;
    privaet const int WM_CHANGECBCHAIN = 0x030D;

    你还需要一个IntPtr变量来保存下一个监听器的句柄: 隐藏,复制Code

    private IntPtr _NextClipboardViewer; 

    你需要为互操作服务添加一个using语句: 隐藏,复制Code

    using System.Runtime.InteropServices; 

    接下来你需要设置PInvoke方法,你需要设置剪贴板监听器: 隐藏,复制Code

    [DllImport("User32.dll"), CharSet = CharSet.Auto)]]
    private static extern IntPtr SetClipboardViewer(IntPtr newviewer);  
     
    [DllImport("User32.dll"), CharSet = CharSet.Auto)] 
    private static extern IntPtr ChangeClipboardChain(IntPtr hwnd, IntPtr newviewer);
     
    [DllImport("User32.dll"), CharSet = CharSet.Auto)]
    private static extern IntPtr SendMessage(IntPtr hwnd, int m, IntPtr wParam, IntPtr lParam); 

    SetClipboardViewer将把给定的句柄插入到剪贴板警告链中,并在一行中返回下一个查看器的句柄。我们将使用这个方法来设置 您的表单作为侦听器,我建议在Form_Load 事件中调用它。 隐藏,复制Code

    private void Form_Load(object sender, EventArgs e)
    {
        _NextClipboardViewer = SetClipboardViewer(this.Handle);
    }

    ChangeClipboardChain将从链中删除给定的句柄,并将最后一个查看器连接到下一个指定的查看器。 当侦听器窗体关闭时,我们将使用它。如果您不处理Form_Closing事件,您可以在这里这样做。如果你是,你有一些条件可以消去 表单关闭,我建议使用Form_Closed中的这个方法。这样,我们只有在关闭表单时才会分离。 隐藏,复制Code

    private void Form_Closed(object sender, FormClosedEventArgs e)
    {
        ChangeClipboardChain(this.Handle, _NextClipboardViewer);
    }

    SendMessage将发送一条windows消息。我们将在拦截消息时使用它。一旦我们采取行动,我们就会会把消息传下去。 所以现在我们需要覆盖WndProc(ref m);方法。如果传入的消息与我们定义的两种类型中的一种匹配,我们希望处理它们,并将它们传递下去: 隐藏,复制Code

    protected override void WndProc(ref m)
    { 
     switch(m.Msg) 
     {
        case WM_DRAWCLIPBOARD:
            // here a change has been made in the contents of clipboard, lets evaluate
            pastebutton.Enable = Clipboard.ContainData(typeof(ClipboardHealthScoreCollection).FullName));
            SendMessage(_NextClipboardViewer, m.Msg, m.WParam, m.LParam);
            break;
        case WM_CHANGECBCHAIN: 
            // here a change has been made in the viewer chain
            if (m.WParam == _NextClipboardViewer) _NextClipboardViewer = m.LParam;
            else SendMessage(_NextClipboardViewer, m.Msg, m.WParam, m.LParam);
            break;
        default:
            base.WndProc(m);  
            break;
     }
    }

    本文转载于:http://www.diyabc.com/frontweb/news266.html

  • 相关阅读:
    input框限制0开头的数字(0除外)
    圆角头像----CSS3特效
    html中div获取焦点,去掉input div等获取焦点时候的边框
    一些常用的html css整理--文本长度截取
    html5本地存储
    div块级元素获取焦点
    Intellij IDEA 搜索文件内容
    web安全漏洞防护
    Intellij IDEA 自动生成 serialVersionUID
    mysql 年龄计算(根据生日)
  • 原文地址:https://www.cnblogs.com/Dincat/p/13437317.html
Copyright © 2020-2023  润新知