简述: 目录树组件QTreeWidget
和停靠区域组件QDockWidget
的讲解和使用
[TOC]
本文初发于 “偕臧的小站“,同步转载于此。
编程环境: win10 x64 专业版 1803
编程软件: Qt Creator 4.8.2 (Enterprise)
, Qt 5.9.8
QTreeWidget属性:
QTreeWidget
类是创建和管理目录树结构的类;
QTreeWidget的每个节点都是一个QTreeWidgetltem对象,添加一个节点前需先创建它,并做
好相关设置。
创建节点的语句是:
item=new QTreeWidgetItem(Mainwindow::itTopItem);
QTreeWidgetltem的 **setlcon()和setText()**都需要传递一个列号作为参数,指定对哪个列进行设置。列号可以直接用数字,但是为了便于理解代码和统一修改,在MainWindow里定义了枚举类型treeColNum,colltem表示第1列,colltemlype表示第2列。setFlags0函数设置节点的一些属性标记,是Qt:ltemFlag枚举类型常量的组合。setData()函数为节点的某一列设置一个角色数据,setData()函数原型为:
void QTreewidgetItem::setData(int column,int role,const QVariant &value)
其中,column是列号,role是角色的值,value是一个QVariant类型的数。
代码中设置节点数据的语句是:
item->setData(Mainwindow::colItem,Qt::UserRole,QVariant(dataStr));
它为节点的第1列,角色Qt:UserRole,设置了一个字符串数据dataStr。Qt:UserRole是枚举类型Qt::ItemDataRole中一个预定义的值,关于节点的角色和Qt:ItemDataRole在后面文章详细介绍。
创建并设置好节点后,用QTreeWidget::addTopLevelltem()函数将节点作为顶层节点添加到目录树。
QDockWidget 属性:
QDockWidget
是可以在QMainWindow 窗口停靠,或在桌面最上层浮动的界面组件。本实例将一个
QTreeWidget组件放置在QDockWidget
区域上,设置其可以在主窗口的左或右侧停靠,也可以浮动。
主要属性如下:
- allowedAreas属性,设置允许停靠区域。由函数setAllowedAreas(Qt::DockWidgetAreas areas)设置允许停靠区,参数areas是枚举类型
- Qt:DockWidgetArea的值的组合,可以设置在窗口的左、右、顶、底停靠,或所有区域都可停靠,
或不允许停靠。
本实例设置为允许左侧和右侧停靠。 - features属性,设置停靠区组件的特性。由setFeatures(Dock WidgetFeatures features)函数设置停靠区组件的特性,参数features是枚举类型QDockWidget::DockWidgetFeature的值的组合,枚举值如下。
QDockWidget::DockWidgetClosable:停靠区可关闭。
QDockWidget:DockWidgetMovable:停靠区可移动。
QDock Widget:DockWidgetFloatable:停靠区可浮动。
QDock Widget:Dock Widget VerticalTitleBar:在停靠区左侧显示垂直标题栏。
QDock Widget::AlDockWidgetFeatures:使用以上所有特征。
QDock Widget::NoDockWidgetFeatures:不能停靠、移动和关闭。
ScrollArea属性:
ScrollArea
是一个提供了一个滚动视图到另一个部件。
滚动区域用于显示一个画面中的子部件的内容。如果部件超过画面的大小,视图可以提供滚动条,这样就都可以看到部件的整个区域。
细节注意:
在主窗口构造函数里将ScrollArea组件设置为主窗口工作区的中心组件后,DockWidget
与ScrollArea
之间自动出现分割条,可以分割两个组件的大小。
部分源码:
其中核心部分的源码,功能实现如下:
其中ExQTreeWidget.h:
#ifndef EXQTREEWIDGET_H
#define EXQTREEWIDGET_H
#include <QMainWindow>
#include <QLabel>
#include <QTreeWidgetItem>
#include <QFileDialog>
namespace Ui {
class ExQTreeWidget;
}
class ExQTreeWidget : public QMainWindow
{
Q_OBJECT
public:
enum treeItemType { //枚举,节点类型
itemRoot,
itemFile,
itemImage
};
enum treeColNum { //目录树列表的编号
colItem = 0,
colItemType = 1
};
explicit ExQTreeWidget(QWidget *parent = nullptr);
~ExQTreeWidget();
void initTree(); //初始化根节点(唯一)
void addFolderItem(QTreeWidgetItem *parItem, QString dirName); //添加目录
void addImageItem(QTreeWidgetItem *parItem, QString fileName); //添加图片文件
QString getFinalFolderName(const QString &pathName); //从完整的路径里面,获取最后的文件夹名称
void changeItemCaption(QTreeWidgetItem* parItem); //遍历item下面的所有节点
void displayImage(QTreeWidgetItem* item); //显示当前item的图片(默认以适配高度)
private slots:
void on_actAddFolder_triggered(); //增加文件夹
void on_actAddFile_triggered(); //添加图片文件
void on_actDeleFile_triggered(); //删除节点
void on_actScanItems_triggered(); //遍历所有的顶层节点(本处只有一个root顶层节点)
void on_actAdaptiveHeight_triggered(); //图片自动适应高度
void on_actAdaptiveWidth_triggered(); //图片自动适应宽度
void on_actAmplification_triggered(); //放大
void on_actShrink_triggered(); //缩小
void on_actZoomRealSize_triggered(); //还原
void on_actDockFloating_triggered(bool check); //设置Dock窗口是否浮动
void on_actDockVisible_triggered(bool checked); //设置Dock窗口是否隐藏不显示
void on_actQiut_triggered(); //退出
void on_treeFiles_currentItemChanged(QTreeWidgetItem *current, QTreeWidgetItem *previous); //当前节点变化的时候,自动加载当前图片
void on_dockWidget_visibilityChanged(bool visible); //单击DockWidget组件的标题栏的关闭按钮时候,会隐藏在停靠区域,并且发射信号visibilityChanged; 停靠区域可见性变化
void on_dockWidget_topLevelChanged(bool topLevel); //当拖动DockWidget组件,使其浮动或者停靠时候,会发射信号topLevelChanged; 更新其Action的状态
private:
Ui::ExQTreeWidget *ui;
QLabel *m_labFlie; //状态栏显示当前文件路径
QPixmap m_curPixmap; //显示当前文件图片
float m_ratio; //图片缩放比例
};
#endif // EXQTREEWIDGET_H
和ExQTreeWidget.cpp:如下
#include "ExQTreeWidget.h"
#include "ui_ExQTreeWidget.h"
ExQTreeWidget::ExQTreeWidget(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::ExQTreeWidget)
{
ui->setupUi(this);
setWindowTitle(QObject::tr("QTreeWidget和QDockWidget的讲解和使用"));
setCentralWidget(ui->scrollArea); //设置scrollArea为中心控件
initTree();
m_labFlie = new QLabel("当前文件的路径:", this);
ui->statusBar->addWidget(m_labFlie);
}
ExQTreeWidget::~ExQTreeWidget()
{
delete ui;
}
//+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//初始化根节点(只能够有唯一)
void ExQTreeWidget::initTree()
{
//准备工作
ui->treeFiles->clear();
QString dataStr = "";
QIcon icon;
icon.addFile(":/image/Image001.jpg");
//创建唯一root的节点
QTreeWidgetItem* root = new QTreeWidgetItem(treeItemType::itemRoot);
root->setIcon(treeColNum::colItem, icon);
root->setText(treeColNum::colItem, QString("相簿"));
root->setText(treeColNum::colItemType, QString("treeItemType"));
root->setFlags(Qt::ItemIsSelectable | Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsAutoTristate);
root->setCheckState(treeColNum::colItem, Qt::Unchecked);
root->setData(treeColNum::colItem, Qt::UserRole, QVariant(dataStr));
//添加顶层节点
ui->treeFiles->addTopLevelItem(root);
}
//添加目录节点
void ExQTreeWidget::addFolderItem(QTreeWidgetItem *parItem, QString dirName)
{
QIcon icon;
icon.addFile(":/image/Image006.jpg");
//添加一个新的节点
QTreeWidgetItem* item = new QTreeWidgetItem(treeItemType::itemFile);
QString folderName = getFinalFolderName(dirName);
item->setIcon(treeColNum::colItem, icon);
item->setText(treeColNum::colItem, folderName);
item->setText(treeColNum::colItemType, QString("treeItemType"));
item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsAutoTristate);
item->setCheckState(treeColNum::colItem, Qt::Unchecked);
item->setData(treeColNum::colItem, Qt::UserRole, QVariant(dirName));
//添加子节点
if (parItem->type() == treeItemType::itemFile) { //若是文件节点
parItem->addChild(item);
} else if (parItem->type() == treeItemType::itemRoot) { //若是唯一root节点
QTreeWidgetItem *root = ui->treeFiles->topLevelItem(0);
root->addChild(item);
}
}
//添加图片节点
void ExQTreeWidget::addImageItem(QTreeWidgetItem *parItem, QString fileName)
{
if (parItem == nullptr)
return;
QIcon icon;
icon.addFile(":/image/Image014.jpg");
//添加一个新的节点
QTreeWidgetItem* item = new QTreeWidgetItem(treeItemType::itemImage);
QString folderName = getFinalFolderName(fileName);
item->setIcon(treeColNum::colItem, icon);
item->setText(treeColNum::colItem, folderName);
item->setText(treeColNum::colItemType, QString("treeItemType"));
item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsAutoTristate);
item->setCheckState(treeColNum::colItem, Qt::Unchecked);
item->setData(treeColNum::colItem, Qt::UserRole, QVariant(fileName));
//添加子节点
if (parItem->type() == treeItemType::itemFile) { //若是文件节点
parItem->addChild(item);
} else if (parItem->type() == treeItemType::itemRoot) { //若是唯一root节点
QTreeWidgetItem *root = ui->treeFiles->topLevelItem(0);
root->addChild(item);
}
}
//从完整的路径里面,获取最后的文件夹名称
QString ExQTreeWidget::getFinalFolderName(const QString &pathName)
{
QString path = pathName;
int cnt = pathName.count();
int i = pathName.lastIndexOf("/");
QString str = pathName.right(cnt - i - 1);
return str;
}
//遍历传进来的父节点下的所有子节点;每遍历过该节点,就在其节点的信息加一个#
void ExQTreeWidget::changeItemCaption(QTreeWidgetItem *parItem)
{
QString str = "# " + parItem->text(treeColNum::colItem);
parItem->setText(treeColNum::colItem, str);
if (parItem->childCount() < 0)
return;
for (int i = 0; i < parItem->childCount(); i++) {
changeItemCaption(parItem->child(i)); //回调,调用自己
}
}
//显示当前item的图片(默认以适配高度)
void ExQTreeWidget::displayImage(QTreeWidgetItem *item)
{
QString fileName = item->data(treeColNum::colItem, Qt::UserRole).toString();
m_labFlie->setText(fileName);
m_curPixmap.load(fileName); //从文件载入图片
on_actAdaptiveHeight_triggered(); //自动适应高度显示
ui->actAmplification->setEnabled(true);
ui->actShrink->setEnabled(true);
ui->actZoomRealSize->setEnabled(true);
ui->actAdaptiveHeight->setEnabled(true);
ui->actAdaptiveWidth->setEnabled(true);
}
//+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
//增加文件夹
void ExQTreeWidget::on_actAddFolder_triggered()
{
QString path = QFileDialog::getExistingDirectory(); //选择目录
if (! path.isEmpty()) {
QTreeWidgetItem* item = ui->treeFiles->currentItem(); //获取当前节点
if(item != nullptr)
addFolderItem(item, path);
}
}
//添加图片
void ExQTreeWidget::on_actAddFile_triggered()
{
QStringList list = QFileDialog::getOpenFileNames(this, "选择多个将要加载的图片", "", "Images(*.jpg, *.png, *.*)"); //选择目录
if (! list.isEmpty()) {
QTreeWidgetItem* parItem = nullptr;
QTreeWidgetItem* item = ui->treeFiles->currentItem(); //获取当前节点
if (item == nullptr)
return;
if (item->type() == treeItemType::itemImage) { //获得父节点
parItem = item->parent();
} else {
parItem = item;
}
for (int i = 0; i < list.size(); i++) {
QString strName = list.at(i); //获得文件名称
addImageItem(parItem, strName); //添加图片文件到文件节点
}
}
}
//删除节点
void ExQTreeWidget::on_actDeleFile_triggered()
{
QTreeWidgetItem* parItem = nullptr;
QTreeWidgetItem* currItem = ui->treeFiles->currentItem();
if (currItem->type() != treeItemType::itemRoot)
parItem = currItem->parent(); //只能够由其父节点删除
// else
// ui->treeFiles->takeTopLevelItem(0); //删除顶层节点使用这个
if (currItem == nullptr || parItem == nullptr)
return;
parItem->removeChild(currItem); //移除没有从内存中删除,所以delete删除
delete currItem;
}
//遍历所有的顶层节点(本处只有一个root顶层节点)
void ExQTreeWidget::on_actScanItems_triggered()
{
for (int i = 0; i < ui->treeFiles->topLevelItemCount(); i++) {
QTreeWidgetItem* currItem = ui->treeFiles->topLevelItem(i); //顶层item
changeItemCaption(currItem);
}
}
//图片自动适应高度
void ExQTreeWidget::on_actAdaptiveHeight_triggered()
{
int height = ui->scrollArea->height(); //得到scrollArea的高度
int realHeight = m_curPixmap.height(); //原始图片的实际高度
m_ratio = height * 1.0 / realHeight; //当前显示比例,必须转换为浮点数
QPixmap pixmap = m_curPixmap.scaledToHeight(height - 50); //图片缩放到指定高度
ui->labDisplay->setPixmap(pixmap); //设置Label的PixMap
}
//图片自动适应宽度
void ExQTreeWidget::on_actAdaptiveWidth_triggered()
{
int width = ui->scrollArea->width();
int realWidth = m_curPixmap.width();
m_ratio = width * 1.0 / realWidth;
QPixmap pixmap = m_curPixmap.scaledToHeight(width - 50);
ui->labDisplay->setPixmap(pixmap);
}
//当前节点变化的时候,自动加载当前图片
void ExQTreeWidget::on_treeFiles_currentItemChanged(QTreeWidgetItem *current, QTreeWidgetItem *previous)
{
if (current != nullptr && previous != nullptr) {
displayImage(current);
}
}
//放大
void ExQTreeWidget::on_actAmplification_triggered()
{
m_ratio *= 1.2; //在当前比例基础上乘以0.8
int height = m_curPixmap.height() * m_ratio; // 显示宽度
int widht = m_curPixmap.width() * m_ratio; // 显示宽度
QPixmap pix = m_curPixmap.scaled(widht, height); //图片缩放到指定高度和宽度,保持长宽比例
ui->labDisplay->setPixmap(pix);
}
//缩小
void ExQTreeWidget::on_actShrink_triggered()
{
m_ratio *= 0.8;
int height = m_curPixmap.height() * m_ratio;
int widht = m_curPixmap.width() * m_ratio;
QPixmap pix = m_curPixmap.scaled(widht, height);
ui->labDisplay->setPixmap(pix);
}
//还原
void ExQTreeWidget::on_actZoomRealSize_triggered()
{
m_ratio = 1;
int height = m_curPixmap.height();
int widht = m_curPixmap.width();
QPixmap pix = m_curPixmap.scaled(widht, height);
ui->labDisplay->setPixmap(pix);
}
//设置Dock窗口是否浮动
void ExQTreeWidget::on_actDockFloating_triggered(bool check)
{
ui->dockWidget->setFloating(check);
}
//设置Dock窗口是否隐藏不显示
void ExQTreeWidget::on_actDockVisible_triggered(bool checked)
{
ui->dockWidget->setVisible(!checked);
}
//退出
void ExQTreeWidget::on_actQiut_triggered()
{
close();
}
//单击DockWidget组件的标题栏的关闭按钮时候,会隐藏在停靠区域,并且发射信号visibilityChanged; 停靠区域可见性变化
void ExQTreeWidget::on_dockWidget_visibilityChanged(bool visible)
{
ui->actDockVisible->setChecked(visible);
}
//当拖动DockWidget组件,使其浮动或者停靠时候,会发射信号topLevelChanged; 更新其Action的状态
void ExQTreeWidget::on_dockWidget_topLevelChanged(bool topLevel)
{
ui->actDockFloating->setChecked(topLevel);
}
源码下载:
https://github.com/xmuli/QtExamples 【QtQTreeWidgetEx】
Deam运行效果:
先上最终的效果图: