Qt实现不同Treewidget之间拖拽

拖拽是编程中经常要用到的,我这里主要是实习了Treewidget之间直接拖拽Item,按下Ctrl键的话是copy,不按Ctrl则是Move。以下是实现代码

[cpp] view plain copy

  1. class TreeItemMimeData:public QMimeData
  2. {
  3. Q_OBJECT
  4. public:
  5. TreeItemMimeData():QMimeData()
  6. {
  7. m_pDragItem = NULL;
  8. }
  9. ~TreeItemMimeData()
  10. {
  11. }
  12. void SetDragData(QString mimeType , QTreeWidgetItem *pItem)
  13. {
  14. m_format<<mimeType;
  15. m_pDragItem = pItem;
  16. }
  17. QStringList formats() const
  18. {
  19. return m_format;
  20. }
  21. const QTreeWidgetItem* DragItemData() const
  22. {
  23. return m_pDragItem;
  24. }
  25. protected:
  26. QVariant retrieveData(const QString &mimetype, QVariant::Type preferredType) const
  27. {
  28. if (mimetype == "ItemMimeData")
  29. {
  30. return m_pDragItem;
  31. }
  32. else
  33. {
  34. return QMimeData::retrieveData(mimetype, preferredType);
  35. }
  36. }
  37. private:
  38. const QTreeWidgetItem   *m_pDragItem;
  39. QStringList              m_format;
  40. };
  41. class MyTreeWidget:public QTreeWidget
  42. {
  43. Q_OBJECT
  44. public:
  45. MyTreeWidget(QWidget *parent = NULL);
  46. ~MyTreeWidget();
  47. protected:
  48. void mouseMoveEvent(QMouseEvent *event);
  49. void mousePressEvent(QMouseEvent *event);
  50. void mouseReleaseEvent(QMouseEvent *event);
  51. void dragEnterEvent(QDragEnterEvent *event);
  52. void dragMoveEvent(QDragMoveEvent *event);
  53. void dropEvent(QDropEvent *event);
  54. void keyPressEvent(QKeyEvent *event);
  55. void keyReleaseEvent(QKeyEvent *event);
  56. private:
  57. QPoint     m_startDragPoint;
  58. void       performDrag();
  59. bool       m_CtrlPressed;
  60. };

[cpp] view plain copy

  1. MyTreeWidget::MyTreeWidget(QWidget *parent /*= NULL*/):QTreeWidget(parent)
  2. {
  3. m_CtrlPressed = false;
  4. this->setSelectionMode(QAbstractItemView::ExtendedSelection);
  5. this->setAcceptDrops(true);
  6. this->setDragEnabled(true);
  7. this->setStyleSheet("QTreeWidget::item:selected{ background-color: rgb(150, 0, 0)}");
  8. }
  9. MyTreeWidget::~MyTreeWidget()
  10. {
  11. }
  12. void MyTreeWidget::keyPressEvent(QKeyEvent *event)
  13. {
  14. if (event->key() == Qt::Key_Control)
  15. {
  16. m_CtrlPressed = true;
  17. }
  18. }
  19. void MyTreeWidget::keyReleaseEvent(QKeyEvent *event)
  20. {
  21. if (event->key() == Qt::Key_Control)
  22. {
  23. m_CtrlPressed = false;
  24. }
  25. }
  26. void MyTreeWidget::mousePressEvent(QMouseEvent *event)
  27. {
  28. if (event->button() == Qt::LeftButton)
  29. {
  30. m_startDragPoint = event->pos();
  31. }
  32. QTreeWidget::mousePressEvent(event);
  33. }
  34. void MyTreeWidget::mouseMoveEvent(QMouseEvent *event)
  35. {
  36. if (event->buttons() & Qt::LeftButton)
  37. {
  38. int dragDistance = (event->pos() - m_startDragPoint).manhattanLength();
  39. if (dragDistance > QApplication::startDragDistance())
  40. {
  41. performDrag();
  42. }
  43. }
  44. QTreeWidget::mouseMoveEvent(event);
  45. }
  46. void MyTreeWidget::mouseReleaseEvent(QMouseEvent *event)
  47. {
  48. QTreeWidget::mouseReleaseEvent(event);
  49. }
  50. void MyTreeWidget::dragEnterEvent(QDragEnterEvent *event)
  51. {
  52. QWidget *source =  qobject_cast<MyTreeWidget *>(event->source());
  53. if (source /*&& source != this*/)
  54. {
  55. if (m_CtrlPressed)
  56. {
  57. event->setDropAction(Qt::CopyAction);
  58. }
  59. else
  60. {
  61. event->setDropAction(Qt::MoveAction);
  62. }
  63. /*event->setDropAction(Qt::MoveAction);  */
  64. event->accept();
  65. }
  66. }
  67. void MyTreeWidget::dragMoveEvent(QDragMoveEvent *event)
  68. {
  69. QWidget *source =  qobject_cast<MyTreeWidget *>(event->source());
  70. if (source /*&& source != this*/)
  71. {
  72. const TreeItemMimeData *pMimeData = (const TreeItemMimeData *)(event->mimeData());
  73. const QTreeWidgetItem *item = pMimeData->DragItemData();
  74. QTreeWidgetItem *currentItem = this->itemAt(event->pos());
  75. if (currentItem == item)           //不允许拖回到原来的item
  76. {
  77. event->ignore();
  78. }
  79. else
  80. {
  81. setCurrentItem(currentItem);
  82. if (m_CtrlPressed)
  83. {
  84. event->setDropAction(Qt::CopyAction);
  85. }
  86. else
  87. {
  88. event->setDropAction(Qt::MoveAction);
  89. }
  90. //event->setDropAction(Qt::MoveAction);
  91. event->accept();
  92. }
  93. }
  94. }
  95. void MyTreeWidget::dropEvent(QDropEvent *event)
  96. {
  97. QWidget *source =  qobject_cast<MyTreeWidget *>(event->source());
  98. const TreeItemMimeData *pMimeData = (const TreeItemMimeData *)(event->mimeData());
  99. if (source /*&& source != this*/)
  100. {
  101. const QTreeWidgetItem *item = pMimeData->DragItemData();
  102. QTreeWidgetItem *pItem = item->clone();
  103. QTreeWidgetItem *currentItem = this->itemAt(event->pos());
  104. if (currentItem && (currentItem != item))
  105. {
  106. currentItem->addChild(pItem);
  107. }
  108. else
  109. {
  110. this->addTopLevelItem(pItem);
  111. }
  112. if (m_CtrlPressed)
  113. {
  114. event->setDropAction(Qt::CopyAction);
  115. }
  116. else
  117. {
  118. event->setDropAction(Qt::MoveAction);
  119. }
  120. //event->setDropAction(Qt::MoveAction);
  121. event->accept();
  122. }
  123. }
  124. void MyTreeWidget::performDrag()
  125. {
  126. QTreeWidgetItem *item = currentItem();
  127. if (item)
  128. {
  129. TreeItemMimeData *mimeData = new TreeItemMimeData;
  130. mimeData->SetDragData("ItemMimeData",item);
  131. QDrag *drag = new QDrag(this);
  132. drag->setMimeData(mimeData);
  133. drag->setPixmap(QPixmap(":/DragDropDemo/Resources/Mail.png"));
  134. if (m_CtrlPressed)
  135. {
  136. drag->exec(Qt::CopyAction);
  137. }
  138. else
  139. {
  140. drag->exec(Qt::MoveAction);
  141. delete item;
  142. }
  143. }
  144. }

以下是我演示的截图

http://blog.csdn.net/hai200501019/article/details/9322329

时间: 2024-10-13 01:12:04

Qt实现不同Treewidget之间拖拽的相关文章

Qt之QAbstractItemView视图项拖拽(二)

一.需求说明 上一篇文章Qt之QAbstractItemView视图项拖拽(一)讲述了实现QAbstractItemView视图项拖拽的一种方式,是基于QDrag实现的,这个类是qt自己封装好了的,所以可定制性也就没有了那么强,最明显的是,这个类在执行exec方法后,mouse系列的回调接口就被阻塞了,随之而来的问题就是拖拽时item项没有了hover特性,为了解决这个问题,我们就不能使用QDrag类来实现拖拽了,这也是这篇文章我要讲述的内容. 二.效果展示 如图1是demo的效果展示,比较丑,

QT笔记之自定义窗口拖拽移动

1.QT自定义标题栏,拖拽标题栏移动窗口(只能拖拽标题,其他位置无法拖拽) 方法一: 转载:http://blog.sina.com.cn/s/blog_4ba5b45e0102e83h.html .h文件中 1 //自己重新实现拖动操作 2 protected: 3 4 void mouseMoveEvent ( QMouseEvent * event ); 5 6 void mousePressEvent ( QMouseEvent * event ); 7 8 void mouseRele

Qt窗口添加鼠标移动拖拽事件

1. .h文件中添加 private:    QPoint dragPosition; 2. 在cpp文件中重写鼠标点击和拖拽函数 void ShapeWidget::mousePressEvent(QMouseEvent * event){    if (event->button() == Qt::LeftButton) //点击左边鼠标    {         dragPosition = event->globalPos() - frameGeometry().topLeft(); 

qt 拖拽 修改大小

写次篇文章之前,qt窗口的放大缩小和拖拽我都是通过setGeometry方法实现的,但是作为windows程序,windows支持橡 皮筋式(拖拽时有一个虚框)拖拽和拉伸.通过setGeometry方式实现功能是没有这种效果,幸好qt5中提供了一个本地事件处理接口 nativeEvent,具体功能可以看帮助文档,本文只讲述用该接口实现窗口放大.缩小和拖拽,具体实现代码如下: 1 virtual bool nativeEvent(const QByteArray &, void *, long *

qt 拖拽 修改大小(二)

最近项目需要实现windows下橡皮筋的效果,所以对此做了一些了解,特此记录. 首先windows系统是支持橡皮筋效果的,需要使用win32方 法:SystemParametersInfo(SPI_SETDRAGFULLWINDOWS, showFullWindow, NULL, 0);showFullWindow是一个变量,如果需要windows默认支持橡皮筋则需要传递参数false,否则传递参数true,如果使用 windows默认的橡皮筋缩放,效果如图1所示,会产生一个矩形框,不管是窗口移

两个GridView之间数据转移,交互,实现拖拽,网易新闻列表效果实现

两个GridView之间数据转移,交互,实现拖拽,网易新闻列表效果实现 摘要 :android 高仿频道管理网易新闻. 新闻频道增删,排序,以及一些动画的实现 可拖动的GridView 地址  :  http://www.itnose.net/detail/6035345.html

Qt 无边框拖拽实现

头文件定义: ------------------------------------------------------------------ class TDragProxy:public QObject{ Q_OBJECT public:    TDragProxy(QWidget* parent);     ~TDragProxy(); protected:     enum WidgetRegion     {         Top = 0,         TopRight,  

Android 仿今日头条频道管理(下)(GridView之间Item的移动和拖拽)

前言 上篇博客我们说到了今日头条频道管理的操作交互体验,我也介绍了2个GridView之间Item的相互移动.详情请參考:Android 仿今日头条频道管理(上)(GridView之间Item的移动和拖拽) 今天把相对照较复杂的gridView的拖拽也记录下.在開始之前我们事先要了解下Android的事件分发机制.网上这方面的资料也比較多.由于自己定义控件大部分要用到事件分发机制的知识. 实现思路 要实现Item的拖拽.事实上并非真正要去拖拽GridView的Item.而是使用WindowMan

Extjs4 实现两个DataView之间元素的拖拽添加及删除

最近项目接到一个需求,要求用拖拽实现在两个Panel之间实现拖拽添加和删除元素的功能. 首先想到的是EXTJS提供的View组件,View组件绑定一个Store和Template就可以得到预期的UI显示效果,再加上EXTJS提供的DD(Drag and Drop)功能,则可以实现两个View组件之前的元素拖拽添加以及删除. 效果如下: Demo代码实例如下: 1 Ext.onReady(function(){ 2 3 var columnData = [ 4 ["Consignee",