Qt:在TreeModel+QTreeView中使用复选框(checkbox)

·转 http://blog.csdn.net/vah101/article/details/6190809

在QT的TreeView中,能够使用复选框,并且选中父节点的复选框可以全选或取消子节点的复选框。这里就以QT附带的simpletreemodel项目为例,说明一下其用法。simpletreemodel项目的路径通常在qt目录的example目录的itemviews目录下,例如,我的就在C:/Qt/2010.05/qt/examples/itemviews里。

1.在头文件treemodel.h中,需要增加头文件
#include
#include
然后在treemodel类的定义中,加入setdata函数和m_checkedList变量的定义

public:
   bool setData( const QModelIndex & index, const QVariant & value, int role = Qt::EditRole );
   QList<QPersistentModelIndex> m_checkedList;

其中,setData是treemodel类的父类QAbstractItemModel 中定义的一个函数,它的功能是响应鼠标点击结点的动作。
m_checkedList则是用来保存被选中(复选框内打勾)的结点的信息

2.在到treemodel.cpp文件中修改。主要是修改flags()、data()两个函数,并实现setData()函数。
flags()函数修改为:

Qt::ItemFlags TreeModel::flags(const QModelIndex &index) const
{
   if (!index.isValid())
   return 0;
   if (index.column()==0)   //如果是第一列的结点,则使其有显示checkbox的能力
   return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsUserCheckable;
   return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
}

主要修改的是,当判断出结点的位置位于第一列,则增加Qt :: ItemIsUserCheckable ,使其具备显示checkbox的能力

然后再修改data()函数:

QVariant TreeModel::data(const QModelIndex &index, int role) const
{
   if (!index.isValid())
   return QVariant();
   if (role==Qt::CheckStateRole && index.column()==0) //判断显示的对象是checkbox,并且位于第一列
   {
   if (m_checkedList.contains(index))    //在m_checkedList中查找,如果有,显示checkbox被选中
   return Qt::Checked;
   else
   return Qt::Unchecked;             //如果没有显示checkbox没被选中
   }
   if (role != Qt::DisplayRole)
   return QVariant();
   TreeItem *item = static_cast<TreeItem*>(index.internalPointer());
   return item->data(index.column());
}

最后是实现setData()函数,这个相对来说复杂一些

bool TreeModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
   if (role==Qt::CheckStateRole && index.column()==0)
   {
   if (value==Qt::Unchecked)
   {
   m_checkedList.removeOne(index);
   emit(dataChanged(index, index));
   }
   else if(value==Qt::Checked)
   if (!m_checkedList.contains(index))
   {
   m_checkedList.append(index);
   }
   if (m_checkedList.contains(index))
   {
   m_checkedList.removeOne(index);
   }
   }
   //  int childCount = rowCount();
   int childCount = rowCount(index); 
   if (childCount>0)                    //判断是否有子节点
   {
   for (int i=0;i<childCount;i++)
   {
   QModelIndex child = this->index(i, 0, index); //获得子节点的index
   setData(child, value, Qt::CheckStateRole);    //递归,将子节点的checkbox设为选中状态
   }
}
   return true;
   }
}

主要是判断对checkbox的操作是选中,还是反选中。如果是选中则将该结点的index加入m_checkedList中,并发送dataChanged信号。反之则将该节点的index从m_checkedList中删除,也发送dataChanged信号。dataChanged信号会触发相应的槽函数,并且会调用到data()函数,这样会重新加载这个结点的状态

参考:rex237专栏的http://blog.csdn.net/Rex237/archive/2010/09/09/5873492.aspx

另外,在http://www.qtcn.org/bbs/read.php?tid=28120
有人提到了可以用QStandardItem来实现TreeView中使用checkbox,可以参考一下

QLineEdit如何实现clicked触发全选,全选的槽是
void QLineEdit::selectAll() [slot]