目录
一、QTreeWidget 概述
QTreeWidget是Qt中一个用于显示树形结构数据的控件,它继承自QTreeView,可以显示多列数据和树形结构的层次关系,还提供了许多交互功能。可以支持单选、多选和可编辑的节点,还可以自定义节点的样式和布局。除此之外,QTreeWidget 还支持信号和槽机制,可以方便地处理节点的操作事件,如点击、双击、选择等等。
头文件: #include <QTreeWidget>
qmake: QT += widgets
继承: QTreeView
二、QTreeWidget 的基本使用
2.1、创建 QTreeWidget 控件
使用 QTreeWidget 控件前,首先需要在代码中创建 QTreeWidget 实例。代码如下:
QTreeWidget *tree = new QTreeWidget();
这里创建了一个 QTreeWidget 实例,并将其保存在 tree 指针中。此时,tree 控件还没有设置显示大小和位置,需要设置它们才能正常显示。
2.2、设置 QTreeWidget 的大小和位置
在 Qt 中,可以使用 QWidget::resize() 和 QWidget::move() 函数来设置控件的大小和位置。代码如下:
tree->resize(400, 300); // 设置控件大小为 400 x 300
tree->move(100, 100); // 设置控件在窗口中的位置为 (100, 100)
这里设置了 tree 控件的大小为 400 x 300 像素,并将其移动到窗口的 (100, 100) 像素处。
2.3、设置 QTreeWidget 的列数和列标题
QTreeWidget 可以显示多列数据,可以使用 QTreeWidget::setColumnCount() 和 QTreeWidget::setHeaderLabels() 函数来设置列数和列标题。代码如下:
tree->setColumnCount(2); // 设置列数为 2
tree->setHeaderLabels({"Name", "Value"}); // 设置列标题为 "Name" 和 "Value"
这里设置了 tree 控件的列数为 2,列标题分别为 "Name" 和 "Value"。
2.4、添加节点
QTreeWidget 中的每个节点都是一个 QTreeWidgetItem 实例,可以使用 QTreeWidget::addTopLevelItem()、QTreeWidgetItem::addChild() 和 QTreeWidget::insertTopLevelItem() 函数来添加节点。代码如下:
QTreeWidgetItem *root = new QTreeWidgetItem(tree); // 创建一个根节点
root->setText(0, "Root"); // 设置节点文本
root->setText(1, "0"); // 设置节点文本
tree->addTopLevelItem(root); // 将节点添加到 QTreeWidget 中
QTreeWidgetItem *child1 = new QTreeWidgetItem(root); // 创建一个子节点
child1->setText(0, "Child 1"); // 设置节点文本
child1->setText(1,"10"); // 设置节点文本
root->addChild(child1); // 将节点添加到根节点下
QTreeWidgetItem *child2 = new QTreeWidgetItem(root); // 创建另一个子节点
child2->setText(0, "Child 2"); // 设置节点文本
child2->setText(1, "20"); // 设置节点文本
tree->insertTopLevelItem(0, child2); // 将节点插入到 QTreeWidget 的第一个位置
这里创建了一个根节点 root,其文本为 "Root" 和 "0",并将其添加到 QTreeWidget 中。然后创建了两个子节点 child1 和 child2,分别作为 root 的子节点,并设置其文本。注意,要将子节点添加到父节点中,可以使用 QTreeWidgetItem::addChild() 函数或 QTreeWidget::insertTopLevelItem() 函数。
2.5、读取节点
可以使用 QTreeWidget::topLevelItem() 和 QTreeWidgetItem::child() 函数来读取节点。代码如下:
QTreeWidgetItem *item = tree->topLevelItem(0); // 获取第一个根节点
if (item) {
QString text1 = item->text(0); // 获取节点的第一列文本
QString text2 = item->text(1); // 获取节点的第二列文本
qDebug() << text1 << text2; // 输出节点文本
QTreeWidgetItem *child = item->child(0); // 获取根节点的第一个子节点
if (child) {
QString text3 = child->text(0); // 获取子节点的第一列文本
QString text4 = child->text(1); // 获取子节点的第二列文本
qDebug() << text3 << text4; // 输出子节点文本
}
}
这里先获取了 QTreeWidget 的第一个根节点,并输出其文本。然后获取根节点的第一个子节点,并输出其文本。
2.6、设置节点数据
QTreeWidgetItem 可以存储自定义数据,可以使用 QTreeWidgetItem::setData() 函数来设置节点的数据。代码如下:
QTreeWidgetItem *item = tree->topLevelItem(0); // 获取第一个根节点
if (item) {
item->setData(2, Qt::UserRole, "custom data"); // 设置第三列的自定义数据
}
// 读取节点数据可以使用item->data(2, Qt::UserRole).toString();
这里设置了根节点的第三列数据为 "custom data"。
2.7、自定义节点样式
QTreeWidget 的节点可以根据需求进行自定义样式的设置。可以使用 QTreeWidget::setItemDelegate() 函数来设置 QItemDelegate,从而实现节点样式的自定义。代码如下:
class MyDelegate : public QItemDelegate
{
public:
MyDelegate(QObject *parent = nullptr) : QItemDelegate(parent) {}
void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const override {
QStyleOptionViewItem opt = option;
if (opt.state & QStyle::State_Selected) {
painter->fillRect(opt.rect, opt.palette.highlight());
opt.palette.setColor(QPalette::Text, opt.palette.highlightedText().color());
}
painter->drawText(opt.rect, Qt::AlignVCenter | Qt::AlignLeft | Qt::TextWordWrap, index.data(Qt::DisplayRole).toString());
}
};
// 在创建 QTreeWidget 对象后设置自定义的 QItemDelegate
QTreeWidget *tree = new QTreeWidget(parent);
tree->setItemDelegate(new MyDelegate(tree));
这里定义了一个 MyDelegate 类来继承自 QItemDelegate,重写了其 paint() 函数以实现节点样式的自定义。在创建 QTreeWidget 对象后,使用 QTreeWidget::setItemDelegate() 函数设置自定义的 QItemDelegate。
三、注意事项
1、 QTreeWidget 中的数据存储在 QTreeWidgetItem 中,因此在对 QTreeWidget 进行数据操作时,必须同时对其对应的 QTreeWidgetItem 进行相应的操作。
2、 QTreeWidget 的节点数据的列数默认为 1,可以使用 QTreeWidget::setColumnCount() 函数进行设置。
3、 QTreeWidgetItem 的列数必须与 QTreeWidget 的列数相同,否则节点的数据将无法正确显示。
四、完整示例
#include <QApplication>
#include <QDebug>
#include <QHeaderView>
#include <QItemDelegate>
#include <QPainter>
#include <QTreeWidget>
#include <QTreeWidgetItem>
class MyDelegate : public QItemDelegate
{
public:
MyDelegate(QObject *parent = nullptr) : QItemDelegate(parent) {}
void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const override {
QStyleOptionViewItem opt = option;
if (opt.state & QStyle::State_Selected) {
painter->fillRect(opt.rect, opt.palette.highlight());
opt.palette.setColor(QPalette::Text, opt.palette.highlightedText().color());
}
painter->drawText(opt.rect, Qt::AlignVCenter | Qt::AlignLeft
| Qt::TextWordWrap, index.data(Qt::DisplayRole).toString());
}
};
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
// 创建 QTreeWidget 对象
QTreeWidget *tree = new QTreeWidget;
tree->setColumnCount(3); // 设置列数为 3
tree->setHeaderLabels({"Name", "Value", "Data"}); // 设置列标签
tree->header()->setSectionResizeMode(QHeaderView::ResizeToContents); // 自适应列宽
// 创建根节点
QTreeWidgetItem *root = new QTreeWidgetItem(tree);
root->setText(0, "Root");
root->setText(1, "0");
root->setData(2, Qt::UserRole, "root data");
tree->addTopLevelItem(root);
// 创建子节点 1
QTreeWidgetItem *child1 = new QTreeWidgetItem(root);
child1->setText(0, "Child 1");
child1->setText(1, "10");
root->addChild(child1);
// 创建子节点 2
QTreeWidgetItem *child2 = new QTreeWidgetItem(root);
child2->setText(0, "Child 2");
child2->setText(1, "20");
tree->insertTopLevelItem(0, child2);
// 设置自定义样式
tree->setItemDelegate(new MyDelegate(tree));
// 读取节点数据
QTreeWidgetItem *item = tree->topLevelItem(0);
if (item) {
qDebug() << "Root name: " << item->text(0); // 输出 "Root"
qDebug() << "Root value: " << item->text(1); // 输出 "0"
qDebug() << "Root data: " << item->data(2, Qt::UserRole).toString(); // 输出 "root data"
}
// 显示 QTreeWidget 对象
tree->show();
return app.exec();
}
运行上述示例程序,可以看到一个简单的 QTreeWidget 对象,其中包含三列数据,根节点下有两个子节点。