Qt实现QQ好友下拉列表(用QListView实现,所以还得定义它的Model)

偶然发现Qt有个控件可以实现下拉列表,所以就试着实现一下类似QQ面板的下拉列表,这里主要实现几个功能:

1.可以删除列表中图标

2.可以像qq一样的,把某个分组下的图标转移到另外的分组

3.添加分组

代码里写了注释了,这里就不重复了,下面直接看代码吧。

自定义的数据模型

ListModel继承了QAbstractListModel,主要是实现要显示的数据结构。用的是model/view的三层结构,这样好拆分

[cpp] view plain copy

  1. struct ListItemData
  2. {
  3. QString  iconPath;
  4. QString  Name;
  5. };
  6. class ListModel:public QAbstractListModel
  7. {
  8. Q_OBJECT
  9. public:
  10. ListModel(QObject *parent = NULL);
  11. ~ListModel();
  12. void init();
  13. void addItem(ListItemData *pItem);
  14. QVariant data ( const QModelIndex & index, int role = Qt::DisplayRole ) const ;
  15. int rowCount ( const QModelIndex & parent = QModelIndex() ) const;
  16. void deleteItem(int index);
  17. ListItemData* getItem(int index );
  18. protected:
  19. private:
  20. vector<ListItemData*> m_ItemDataVec;
  21. };

[cpp] view plain copy

  1. <pre name="code" class="cpp">ListModel::ListModel( QObject *parent /*= NULL*/ ):QAbstractListModel(parent)
  2. {
  3. init();
  4. }
  5. ListModel::~ListModel()
  6. {
  7. }
  8. QVariant ListModel::data( const QModelIndex & index, int role /*= Qt::DisplayRole */ ) const
  9. {
  10. if (index.row() > m_ItemDataVec.size())
  11. {
  12. return QVariant();
  13. }
  14. else
  15. {
  16. switch (role)
  17. {
  18. case Qt::DisplayRole:
  19. {
  20. return m_ItemDataVec[index.row()]->Name;
  21. }
  22. break;
  23. case Qt::DecorationRole:
  24. {
  25. return QIcon(m_ItemDataVec[index.row()]->iconPath);
  26. }
  27. break;
  28. case Qt::SizeHintRole:
  29. {
  30. return QSize(10,50);
  31. }
  32. }
  33. }
  34. return QVariant();
  35. }
  36. int ListModel::rowCount( const QModelIndex & parent /*= QModelIndex() */ ) const
  37. {
  38. return m_ItemDataVec.size();
  39. }
  40. void ListModel::init()
  41. {
  42. for (int i = 1; i < 26; ++i)
  43. {
  44. ListItemData *pItem = new ListItemData;
  45. pItem->Name = QString::number(i);
  46. pItem->iconPath = QString(":/QQPanel/Resources/%1.jpg").arg(i);
  47. QFile Iconfile(pItem->iconPath);
  48. if (Iconfile.exists())
  49. {
  50. m_ItemDataVec.push_back(pItem);
  51. }
  52. }
  53. }
  54. void ListModel::deleteItem( int index )
  55. {
  56. vector<ListItemData*>::iterator it = m_ItemDataVec.begin();
  57. m_ItemDataVec.erase(it + index);
  58. }
  59. void ListModel::addItem( ListItemData *pItem )
  60. {
  61. if (pItem)
  62. {
  63. this->beginInsertRows(QModelIndex(),m_ItemDataVec.size(),m_ItemDataVec.size() + 1);
  64. <span style="white-space:pre">      </span>m_ItemDataVec.push_back(pItem);
  65. <span style="white-space:pre">      </span>this->endInsertRows();
  66. }
  67. }
  68. ListItemData* ListModel::getItem( int index )
  69. {
  70. if (index > -1 && index < m_ItemDataVec.size())
  71. {
  72. return m_ItemDataVec[index];
  73. }
  74. }
  75. </pre><br>
  76. <br>
  77. <pre></pre>
  78. <h1><a name="t1"></a><br>
  79. </h1>
  80. <h1><a name="t2"></a>自定义的列表</h1>
  81. <div>这个类才是重点,因为这里实现了删除和转移图标的几个重要的函数。</div>
  82. <pre name="code" class="cpp">class MyListView:public QListView
  83. {
  84. Q_OBJECT
  85. public:
  86. MyListView(QWidget *parent = NULL);
  87. ~MyListView();
  88. void setListMap(map<MyListView*,QString> *pListMap);
  89. void addItem(ListItemData *pItem);
  90. protected:
  91. void contextMenuEvent ( QContextMenuEvent * event );
  92. private slots:
  93. void deleteItemSlot(bool bDelete);
  94. void moveSlot(bool bMove);
  95. private:
  96. int  m_hitIndex;
  97. ListModel   *m_pModel;
  98. ////记录分组和分组名字的映射关系,这个值跟QQPanel类中的映射组的值保持一致
  99. //这里还有一个用处就是在弹出的菜单需要分组的名称
  100. map<MyListView*,QString> *m_pListMap;
  101. //记录每个菜单项对应的列表,才能知道要转移到那个分组
  102. map<QAction*,MyListView*> m_ActionMap;
  103. };</pre><br>
  104. <pre name="code" class="cpp">MyListView::MyListView( QWidget *parent /*= NULL*/ ):QListView(parent)
  105. {
  106. m_hitIndex = -1;
  107. m_pModel = new ListModel;
  108. this->setModel(m_pModel);
  109. m_pListMap = NULL;
  110. }
  111. MyListView::~MyListView()
  112. {
  113. }
  114. void MyListView::contextMenuEvent( QContextMenuEvent * event )
  115. {
  116. int hitIndex = this->indexAt(event->pos()).column();
  117. if (hitIndex > -1)
  118. {
  119. QMenu *pMenu = new QMenu(this);
  120. QAction *pDeleteAct = new QAction(tr("删除"),pMenu);
  121. pMenu->addAction(pDeleteAct);
  122. connect(pDeleteAct,SIGNAL(triggered (bool)),this,SLOT(deleteItemSlot(bool)));
  123. QMenu *pSubMenu = NULL;
  124. map<MyListView*,QString>::iterator it = m_pListMap->begin();
  125. for (it;it != m_pListMap->end();++it)
  126. {
  127. if (!pSubMenu)
  128. {
  129. pSubMenu = new QMenu(tr("转移联系人至") ,pMenu);
  130. pMenu->addMenu(pSubMenu);
  131. }
  132. if (it->first != this)
  133. {
  134. QAction *pMoveAct = new QAction( it->second ,pMenu);
  135. //记录菜单与分组的映射,在moveSlot()响应时需要用到。
  136. m_ActionMap.insert(pair<QAction*,MyListView*>(pMoveAct,it->first));
  137. pSubMenu->addAction(pMoveAct);
  138. connect(pMoveAct,SIGNAL(triggered (bool)),this,SLOT(moveSlot(bool)));
  139. }
  140. }
  141. pMenu->popup(mapToGlobal(event->pos()));
  142. }
  143. }
  144. void MyListView::deleteItemSlot( bool bDelete )
  145. {
  146. int index = this->currentIndex().row();
  147. if (index > -1)
  148. {
  149. m_pModel->deleteItem(index);
  150. }
  151. }
  152. void MyListView::setListMap( map<MyListView*,QString> *pListMap )
  153. {
  154. m_pListMap = pListMap;
  155. }
  156. void MyListView::addItem( ListItemData *pItem )
  157. {
  158. m_pModel->addItem(pItem);
  159. }
  160. void MyListView::moveSlot( bool bMove )
  161. {
  162. QAction *pSender = qobject_cast<QAction*>(sender());
  163. if (pSender)
  164. {
  165. //根据点击的菜单,找到相应的列表,然后才能把图标转移过去
  166. MyListView *pList = m_ActionMap.find(pSender)->second;
  167. if (pList)
  168. {
  169. int index = this->currentIndex().row();
  170. ListItemData *pItem = m_pModel->getItem(index);
  171. pList->addItem(pItem);
  172. //添加到别的分组,就在原来的分组中删除掉了
  173. m_pModel->deleteItem(index);
  174. }
  175. }
  176. //操作完了要把这个临时的映射清空
  177. m_ActionMap.clear();
  178. }
  179. </pre><br>
  180. <h1><a name="t3"></a>自定义的主控件</h1>

[cpp] view plain copy

  1. class QQPanel : public QWidget
  2. {
  3. Q_OBJECT
  4. public:
  5. QQPanel(QWidget *parent = 0, Qt::WFlags flags = 0);
  6. ~QQPanel();
  7. protected:
  8. void contextMenuEvent ( QContextMenuEvent * event );
  9. protected slots:
  10. void addGroupSlot(bool addgroup);
  11. private:
  12. QToolBox    *m_pBox;
  13. map<MyListView*,QString> *m_pListMap;    //记录分组和分组名字的映射关系,好在转移图标时知道转移到那个分组
  14. };

[cpp] view plain copy

  1. QQPanel::QQPanel(QWidget *parent, Qt::WFlags flags)
  2. : QWidget(parent, flags)
  3. {
  4. m_pBox = new QToolBox(this);
  5. m_pListMap = new map<MyListView*,QString>();
  6. MyListView *pListView = new MyListView(this);
  7. pListView->setViewMode(QListView::ListMode);
  8. pListView->setStyleSheet("QListView{icon-size:40px}");
  9. m_pBox->addItem(pListView,tr("我的好友"));
  10. m_pListMap->insert(pair<MyListView*,QString>(pListView,tr("我的好友")));
  11. MyListView *pListView1 = new MyListView(this);
  12. pListView1->setViewMode(QListView::ListMode);
  13. pListView1->setStyleSheet("QListView{icon-size:40px}");
  14. m_pBox->addItem(pListView1,tr("陌生人"));
  15. m_pListMap->insert(pair<MyListView*,QString>(pListView1,tr("陌生人")));
  16. pListView->setListMap(m_pListMap);
  17. pListView1->setListMap(m_pListMap);
  18. m_pBox->setFixedWidth(150);
  19. m_pBox->setMinimumHeight(500);
  20. this->setMinimumSize(200,500);
  21. //ui.setupUi(this);
  22. }
  23. QQPanel::~QQPanel()
  24. {
  25. }
  26. void QQPanel::contextMenuEvent( QContextMenuEvent * event )
  27. {
  28. QMenu *pMenu = new QMenu(this);
  29. QAction *pAddGroupAct = new QAction(tr("添加分组"),pMenu);
  30. pMenu->addAction(pAddGroupAct);
  31. connect(pAddGroupAct,SIGNAL(triggered (bool)),this,SLOT(addGroupSlot(bool)));
  32. pMenu->popup(mapToGlobal(event->pos()));
  33. }
  34. void QQPanel::addGroupSlot( bool addgroup )
  35. {
  36. QString name = QInputDialog::getText(this,tr("输入分组名"),tr(""));
  37. if (!name.isEmpty())
  38. {
  39. MyListView *pListView1 = new MyListView(this);
  40. pListView1->setViewMode(QListView::ListMode);
  41. pListView1->setStyleSheet("QListView{icon-size:40px}");
  42. m_pBox->addItem(pListView1,name);
  43. m_pListMap->insert(pair<MyListView*,QString>(pListView1,name));
  44. }
  45. //要确保每个MyListView钟的m_pListMap都是一致的,不然就会有错了。
  46. //因为弹出的菜单进行转移的时候需要用到
  47. map<MyListView*,QString>::iterator it = m_pListMap->begin();
  48. for (it; it != m_pListMap->end(); ++it)
  49. {
  50. MyListView* pList = it->first;
  51. pList->setListMap(m_pListMap);
  52. }
  53. }

运行结果

由以上两个截图显示,我的好友和陌生人的个有5个图标

以上两个截图显示,把陌生人中图标5转移到我的好友里

以上两个截图,显示添加了一个分组,黑名单,因为我默认列表在创建时都有相同的5个图标

以上三个截图显示了把黑名单里的图标5转移到了我的好友分组里了

当然这个程序算是比较简单的。还不能真正的跟QQ的面板相比,还不能把所有的分组都收起来。以后再慢慢研究怎么实现了,

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

时间: 2024-10-15 10:54:33

Qt实现QQ好友下拉列表(用QListView实现,所以还得定义它的Model)的相关文章

Qt实现 QQ好友列表QToolBox

简述 QToolBox类提供了一个列(选项卡式的)部件条目. QToolBox可以在一个tab列上显示另外一个,并且当前的item显示在当前的tab下面.每个tab都在tab列中有一个索引位置.tab的item是一个QWidget . 简述 详细描述 使用 效果 源码 详细描述 每个item都有一个itemText().一个可选的itemIcon().一个可选的itemToolTip().和一个widget()函数 .item的属性可以通过setItemText().setItemIcon().

基于Qt的类似QQ好友列表抽屉效果的实现

前段时间在忙毕业设计,所以一直没有更新博客.今天答辩完以后,将对我的毕业设计进行模块展示,供Qt初学者进行参考. 毕业设计题目:Linux系统下基于Qt的局域网即时通信系统设计与实现 其中我有一个类似于QQ的好友列表,然后对好友可以进行分组管理,毕设中具体效果图如下: 网上查寻到的设计思路: 1.采用QToolBox的方式,虽然看起来有点样子,但是并不是我们所熟悉的好友列表,比如:http://blog.csdn.net/qianguozheng/article/details/6719074

MapReduce实现QQ好友推荐

大家都知道qq用户量上亿,每个用户又有很多的好友,因此,数据量十分的庞大,如何才能实现QQ的好友推荐呢? 下面举一个例子: A有QQ好友B B有QQ好友C 则A,C有可能是好友. 当A登录的时候,则会向A推荐C,当C登录的时候,则会向C推荐A. Demo 输入数据 map阶段 key:主 value:从 key:从 value:主 将一条记录分别作为key,value进行输出. tom-->jason jason-->tom tom-->lgd lgd-->tom reduce阶段

QQ好友在线/离线,怎么测试?

即时通讯是目前internet上最为流行的通讯方式,各种各样的即时通讯软件也层出不穷,那么今天主要针对QQ好友在线状态/QQ群友在线状态功能出发,一起思考其中的实现原理以及我们如何去测试此功能? 当大家在使用QQ的时候,是否和我一样有如下疑问: 在好友列表中为什么可以实时的看到qq好友的在线.离线等状态? 在QQ群的群友列表中能看到当前群友的在线.离线等状态,是如何实现的? 作为测试工程师究竟该如何去测试里面用到的技术? 我带着这些问题,去搜集了一些资料进行了解.学习和总结后,现分享给大家. 关

javascript社交平台分享-新浪微博、QQ微博、QQ好友、QQ空间、人人网

整理的五个社交平台的分享 <!doctype html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> </head> <body> <h2>分享</h2> <div id="sina">新浪微博</div>

仿QQ好友列表界面的实现

TableView有2种style:UITableViewStylePlain 和 UITableViewStyleGrouped. 但是QQ好友列表的tableView给人的感觉似乎是2个style效果都有,但是tableView不能实现2种效果同时存在. 其实只是用到了Plain这个style,只是在cell的个数显示上做了个处理(个人见解,希望可以帮到有需要的人.....) 当通讯录那一组的cell的组头视图中的button是普通状态下的时候,并不是不显示cell,而是显示一个没有任何内容

java模仿qq好友面板的布局(BoxLayout问题)

.............. JLabel ll = new JLabel(dlg.getNameText() + ":" + dlg.getIPText(), ii[index], JLabel.LEFT); tmp = new JPanel();//将标签添加到这个面板中 tmp.setLayout(new FlowLayout(FlowLayout.CENTER)); tmp.setBackground(new Color(255, 0, 255)); /* * BoxLayou

iOS-QQ好友列表 iOS 页面间几种传值方式(属性,代理,block,单例,通知)

主要是 点击按钮实现下拉 刷新数据 页面间传值 // // HMFriendsModel.h // QQ好友列表 // // Created by YaguangZhu on 15/9/1. // Copyright (c) 2015年 YaguangZhu. All rights reserved. // #import <Foundation/Foundation.h> @interface HMFriendsModel : NSObject @property(nonatomic,cop

QQ好友列表数据模型封装

QQ好友中的信息较多,如果我们单独从plist 中直接取出数据 是可以解决问题 但是相当复杂,以为列表中分组 ,每组中还有不同信息 大致模型是 数组套数组  数组套字典 所以我们要封装数据模型 // // GPGroupController.h // 02-好友分组 // #import <UIKit/UIKit.h> @interface GPGroupController : UIViewController @end // // GPGroupController.m // 02-好友分