前面的例子都是使用的系统提供的拖放对象 QMimeData 进行拖放数据的存储,比如使用 QMimeData::setText() 创建文本,使用 QMimeData::urls() 创建 URL 对象。但是,如果你希望使用一些自定义的对象作为拖放数据,比如自定义类等等,单纯使用 QMimeData 可能就没有那么容易了。为了实现这种操作,我们可以从下面三种实现方式中选择一个:
  1. 将自定义数据作为 QByteArray 对象,使用 QMimeData::setData() 函数作为二进制数据存储到 QMimeData 中,然后使用 QMimeData::Data() 读取;
  2. 继承 QMimeData,重写其中的 formats() 和 retrieveData() 函数操作自定义数据;
  3. 如果拖放操作仅仅发生在同一个应用程序,可以直接继承 QMimeData,然后使用任意合适的数据结构进行存储。
第一种方法不需要继承任何类,但是有一些局限:即是拖放不会发生,我们也必须将自定义的数据对象转换成 QByteArray 对象;如果你希望支持很多种拖放的数据,那么每种类型的数据都必须使用一个 QMimeData 类,这可能会导致类爆炸;如果数据很大的话,这种方式可能会降低系统的可维护性。然而,后两种实现方式就不会有这些问题,或者说是能够减小这种问题,并且能够让我们有完全控制权。
我们先来看一个应用,使用 QTableWidget 来进行拖放操作,拖放的类型包括 plain/text,plain/html 和 plain/csv。如果使用第一种实现方法,我们的代码将会如下所示:
  1. void MyTableWidget::mouseMoveEvent(QMouseEvent *event)
  2. {
  3. if (event->buttons() & Qt::LeftButton) {
  4. int distance = (event->pos() - startPos).manhattanLength();
  5. if (distance >= QApplication::startDragDistance())
  6. performDrag();
  7. }
  8. QTableWidget::mouseMoveEvent(event);
  9. }
  10. void MyTableWidget::performDrag()
  11. {
  12. QString plainText = selectionAsPlainText();
  13. if (plainText.isEmpty())
  14. return;
  15. QMimeData *mimeData = new QMimeData;
  16. mimeData->setText(plainText);
  17. mimeData->setHtml(toHtml(plainText));
  18. mimeData->setData("text/csv", toCsv(plainText).toUtf8());
  19. QDrag *drag = new QDrag(this);
  20. drag->setMimeData(mimeData);
  21. if (drag->exec(Qt::CopyAction | Qt::MoveAction) == Qt::MoveAction)
  22. deleteSelection();
  23. }
对于这段代码,我们应该已经很容易的理解:在 performDrag() 函数中,我们调用 QMimeData 的 setText() 和 setHTML() 函数存储 plain/text 和 plain/html 数据,使用 setData() 将 text/csv 类型的数据作为二进制 QByteArray 类型存储。
  1. QString MyTableWidget::toCsv(const QString &plainText)
  2. {
  3. QString result = plainText;
  4. result.replace("\\", "\\\\");
  5. result.replace("\"", "\\\"");
  6. result.replace("\t", "\", \"");
  7. result.replace("\n", "\"\n\"");
  8. result.prepend("\"");
  9. result.append("\"");
  10. return result;
  11. }
  12. QString MyTableWidget::toHtml(const QString &plainText)
  13. {
  14. QString result = Qt::escape(plainText);
  15. result.replace("\t", "<td>");
  16. result.replace("\n", "\n<tr><td>");
  17. result.prepend("<table>\n<tr><td>");
  18. result.append("\n</table>");
  19. return result;
  20. }
toCsv() 和 toHtml() 函数将数据取出并转换成我们需要的 csv 和 html类型的数据。例如,下面的数据
Red   Green   Blue
Cyan Yellow Magenta
转换成 csv 格式为:
"Red", "Green", "Blue"
"Cyan", "Yellow", "Magenta"
转换成 html 格式为:
<table>
<tr><td>Red<td>Green<td>Blue
<tr><td>Cyan<td>Yellow<td>Magenta
</table>
在放置的函数中我们像以前一样使用:
  1. void MyTableWidget::dropEvent(QDropEvent *event)
  2. {
  3. if (event->mimeData()->hasFormat("text/csv")) {
  4. QByteArray csvData = event->mimeData()->data("text/csv");
  5. QString csvText = QString::fromUtf8(csvData);
  6. // ...
  7. event->acceptProposedAction();
  8. } else if (event->mimeData()->hasFormat("text/plain")) {
  9. QString plainText = event->mimeData()->text();
  10. // ...
  11. event->acceptProposedAction();
  12. }
  13. }
虽然我们接受三种数据类型,但是在这个函数中我们只接受两种类型。至于 html 类型,我们希望如果用户将 QTableWidget 的数据拖到一个 HTML 编辑器,那么它就会自动转换成 html 代码,但是我们不计划支持将外部的 html 代码拖放到 QTableWidget 上。为了让这段代码能够工作,我们需要在构造函数中设置 setAcceptDrops(true) 和 setSelectionMode(ContiguousSelection)。
好了,上面就是我们所说的第一种方式的实现。这里并没有给出完整的实现代码,大家可以根据需要自己实现一下试试。下面我们将按照第二种方法重新实现这个需求。
  1. class TableMimeData : public QMimeData
  2. {
  3. Q_OBJECT
  4. public:
  5. TableMimeData(const QTableWidget *tableWidget,
  6. const QTableWidgetSelectionRange &range);
  7. const QTableWidget *tableWidget() const { return myTableWidget; }
  8. QTableWidgetSelectionRange range() const { return myRange; }
  9. QStringList formats() const;
  10. protected:
  11. QVariant retrieveData(const QString &format,
  12. QVariant::Type preferredType) const;
  13. private:
  14. static QString toHtml(const QString &plainText);
  15. static QString toCsv(const QString &plainText);
  16. QString text(int row, int column) const;
  17. QString rangeAsPlainText() const;
  18. const QTableWidget *myTableWidget;
  19. QTableWidgetSelectionRange myRange;
  20. QStringList myFormats;
  21. };
为了避免存储具体的数据,我们存储 table 和选择区域的坐标的指针。
  1. TableMimeData::TableMimeData(const QTableWidget *tableWidget,
  2. const QTableWidgetSelectionRange &range)
  3. {
  4. myTableWidget = tableWidget;
  5. myRange = range;
  6. myFormats << "text/csv" << "text/html" << "text/plain";
  7. }
  8. QStringList TableMimeData::formats() const
  9. {
  10. return myFormats;
  11. }
构造函数中,我们对私有变量进行初始化。formats() 函数返回的是被 MIME 数据对象支持的数据类型列表。这个列表是没有先后顺序的,但是最佳实践是将“最适合”的类型放在第一位。对于支持多种类型的应用程序而言,有时候会直接选用第一个符合的类型存储。
  1. QVariant TableMimeData::retrieveData(const QString &format,
  2. QVariant::Type preferredType) const
  3. {
  4. if (format == "text/plain") {
  5. return rangeAsPlainText();
  6. } else if (format == "text/csv") {
  7. return toCsv(rangeAsPlainText());
  8. } else if (format == "text/html") {
  9. return toHtml(rangeAsPlainText());
  10. } else {
  11. return QMimeData::retrieveData(format, preferredType);
  12. }
  13. }
 
函数 retrieveData() 将给出的 MIME 类型作为 QVariant 返回。参数 format 的值通常是 formats() 函数返回值之一,但是我们并不能假定一定是这个值之一,因为并不是所有的应用程序都会通过 formats() 函数检查 MIME 类型。一些返回函数,比如 text(), html(), urls(), imageData(), colorData() 和 data() 实际上都是在 QMimeData 的 retrieveData() 函数中实现的。第二个参数 preferredType 给出我们应该在 QVariant 中存储哪种类型的数据。在这里,我们简单的将其忽略了,并且在 else 语句中,我们假定 QMimeData 会自动将其转换成所需要的类型。
  1. void MyTableWidget::dropEvent(QDropEvent *event)
  2. {
  3. const TableMimeData *tableData =
  4. qobject_cast<const TableMimeData *>(event->mimeData());
  5. if (tableData) {
  6. const QTableWidget *otherTable = tableData->tableWidget();
  7. QTableWidgetSelectionRange otherRange = tableData->range();
  8. // ...
  9. event->acceptProposedAction();
  10. } else if (event->mimeData()->hasFormat("text/csv")) {
  11. QByteArray csvData = event->mimeData()->data("text/csv");
  12. QString csvText = QString::fromUtf8(csvData);
  13. // ...
  14. event->acceptProposedAction();
  15. } else if (event->mimeData()->hasFormat("text/plain")) {
  16. QString plainText = event->mimeData()->text();
  17. // ...
  18. event->acceptProposedAction();
  19. }
  20. QTableWidget::mouseMoveEvent(event);
  21. }
在放置的函数中,我们需要按照我们自己定义的数据类型进行选择。我们使用 qobject_cast 宏进行类型转换。如果成功,说明数据来自同一应用程序,因此我们直接设置 QTableWidget 相关 数据,如果转换失败,我们则使用一般的处理方式。

本文出自 “豆子空间” 博客,请务必保留此出处http://devbean.blog.51cto.com/448512/288742

Qt学习之路(54): 自定义拖放数据对象的更多相关文章

  1. Qt 学习之路:自定义事件

    尽管 Qt 已经提供了很多事件,但对于更加千变万化的需求来说,有限的事件都是不够的.例如,我要支持一种新的设备,这个设备提供一种崭新的交互方式,那么,这种事件如何处理呢?所以,允许创建自己的事件 类型 ...

  2. Qt 学习之路 :可视化显示数据库数据

    前面我们用了两个章节介绍了 Qt 提供的两种操作数据库的方法.显然,使用QSqlQuery的方式更灵活,功能更强大,而使用QSqlTableModel则更简单,更方便与 model/view 结合使用 ...

  3. Qt学习之路(45): 自定义model之一

    前面我们说了Qt提供的几个预定义model.但是,面对变化万千的需求,那几个model是远远不能满足我们的需要的.另外,对于Qt这种框架来说,model的选择首先要能满足绝大多数功能的需要,这就是说, ...

  4. Qt学习之路:自定义Model三篇,自定义委托等等

    http://devbean.blog.51cto.com/448512/d-8/p-2

  5. Qt 学习之路 2(53):自定义拖放数据

    Qt 学习之路 2(53):自定义拖放数据 豆子  2013年5月26日  Qt 学习之路 2  13条评论上一章中,我们的例子使用系统提供的拖放对象QMimeData进行拖放数据的存储.比如使用QM ...

  6. Qt 学习之路 2(54):剪贴板

    Qt 学习之路 2(54):剪贴板 豆子 2013年6月8日 Qt 学习之路 2 2条评论 剪贴板的操作经常和前面所说的拖放技术在一起使用.大家对剪贴板都很熟悉.我们可以简单地把它理解成一个数据存储池 ...

  7. Qt 学习之路 2(49):自定义只读模型

    Qt 学习之路 2(49):自定义只读模型 豆子 2013年5月5日 Qt 学习之路 2 18条评论 model/view 模型将数据与视图分割开来,也就是说,我们可以为不同的视图,QListView ...

  8. Qt 学习之路 2(5):自定义信号槽

    Home / Qt 学习之路 2 / Qt 学习之路 2(5):自定义信号槽 Qt 学习之路 2(5):自定义信号槽  豆子  2012年8月24日  Qt 学习之路 2  131条评论 上一节我们详 ...

  9. Qt 学习之路 2(57):可视化显示数据库数据

    Qt 学习之路 2(57):可视化显示数据库数据(skip) 豆子 2013年6月26日 Qt 学习之路 2 26条评论 前面我们用了两个章节介绍了 Qt 提供的两种操作数据库的方法.显然,使用QSq ...

随机推荐

  1. HDU4099(斐波那契数列与字典树)

    题目:Revenge of Fibonacci 题意:给出斐波那契数列的前k位,k不超过40,找出最小的正整数n,满足F(n)的前k位与给定数的前k位相同,斐波那契数列的项数不超过100000. 解析 ...

  2. 禁用Visual Studio 2013的Browser Link功能

    禁用Visual Studio 2013的Browser Link功能 GET http://localhost:37478/7fd25f8af33f443494e765be19be6240/brow ...

  3. the convertion between string and BlobColumn

    It's hard to find some samples about the convertion between string and BlobColumn.AddBlobData. It's ...

  4. java反射机制入门3

    Method对象的机制与实现 1.Method对象概述 1)java.lang.reflect.Method类是用于表示类中.接口中方法对象的类. 2)可以操作类中私有,以及公有等全部方法. 2.Me ...

  5. FtpManager类

    public class FtpManager { /// <summary> /// 主机名 /// </summary> string ftpServerIP; /// & ...

  6. YII2 Gridview

    YII2 Gridview 部分使用规则 1.页面显示的时间戳转换 a. [ 'label'=>'创建日期', 'attribute' => 'created_at', 'filter' ...

  7. C# DES对称加密解密

    /// <summary> /// 加密 /// </summary> /// <param name="str"></param> ...

  8. 2013 南京邀请赛 K题 yet another end of the world

    /** 大意:给定一组x[],y[],z[] 确定有没有两个不同的x[i], x[j] 看是否存在一个ID使得 y[i]<=ID%x[i]<=z[i] y[j]<=ID%x[j]&l ...

  9. 手把手教你图片转ASCII码图

    效果图 基本思路 把图片每个像素点的信息拿出来,最重要的是拿到rgb的值 把每个像素点由rgb转成灰度图像,即0-255 给0-255分级,把每个等级的像素点转换成ascii码,完成 实现 第一步:获 ...

  10. BZOJ 1633: [Usaco2007 Feb]The Cow Lexicon 牛的词典

    题目 1633: [Usaco2007 Feb]The Cow Lexicon 牛的词典 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 401  Solv ...