Qt-MVD模式

一、前言

  MVD 模式的核心价值:关注点分离,让数据管理、显示逻辑和用户交互各自独立,从而创建出更可维护、可扩展的应用程序。

Model: 完全不知道数据会如何显示,只负责存储和管理数据。

View :视图作为桥梁,从 Model 获取数据、将数据传递给 Delegate 进行渲染、处理用户交互事件

Delegate :委托完全控制每个项目的外观,包括颜色、布局、边框等。

好处:

修改方便:

  • 可以轻松更换不同的 Model 来改变数据源

  • 可以轻松更换不同的 View 来改变交互方式

  • 可以轻松更换不同的 Delegate 来改变外观

 松耦合设计

  • 修改数据不影响显示逻辑

  • 修改显示样式不影响数据

  • 各部分可以独立开发和测试

二、Model

 基础模型类型:

QStringListModel - 字符串列表模型

#include <QStringListModel>

QStringListModel *model = new QStringListModel;
model->setStringList({"苹果", "香蕉", "橙子"});

QStandardItemModel - 标准项目模型

#include <QStandardItemModel>

QStandardItemModel *model = new QStandardItemModel;
model->appendRow(new QStandardItem("项目1"));

QFileSystemModel - 文件系统模型

#include <QFileSystemModel>

QFileSystemModel *model = new QFileSystemModel;
model->setRootPath(QDir::homePath());

三、View

  1. QListView:将数据项显示为列表。

  2. QTableView:将数据项显示在表格中。

  3. QTreeView:将数据项显示在层次结构的树中。

  4. QColumnView:使用多个列显示数据项,每个列代表一个层次级别。

  5. QHeaderView:用于显示表格或列表的标题行或标题列,通常不单独使用,而是作为QTableView和QTreeView的一部分。

  6. QListWidgetQTableWidgetQTreeWidget:这些是方便类,内部已经集成了模型,适用于简单应用。但它们不符合严格的MVC模式,因为模型和视图是捆绑在一起的。

四、delegate

常用的委托方式:

  1. 使用内置委托:QT提供了一些内置的委托类,如QStyledItemDelegateQItemDelegate。通常推荐使用QStyledItemDelegate,因为它使用当前样式来绘制项目。 内置委托:

  2. QT的内置委托可以处理大多数常见的数据类型(如字符串、数字、日期等)的显示和编辑。例如,当你在一个模型/视图结构中编辑一个布尔值时,内置委托会提供一个复选框;编辑数字时,会提供一个数字输入框等。

  3. 自定义委托:通过继承QStyledItemDelegateQItemDelegate,并重写其虚函数来实现自定义的显示和编辑行为。

  4.  自定义委托:自定义委托通常需要重写以下一个或多个函数:

  • paint(): 用于自定义项目的显示。

  • sizeHint(): 返回项目的大小。

  • createEditor(): 返回一个用于编辑项目的部件(如QLineEdit、QComboBox等)。

  • setEditorData(): 将模型中的数据设置到编辑器。

  • setModelData(): 将编辑器中的数据保存到模型。

  • updateEditorGeometry(): 设置编辑器在视图中的位置和大小。

五、案例

.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QListView>
#include <QStringListModel>
#include <QStyledItemDelegate>
#include <QPainter>
#include <QVBoxLayout>
#include <QWidget>

// 前向声明
class QListView;
class QStringListModel;
class SimpleDelegate;

QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACE

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = nullptr);
    ~MainWindow();

private:
    Ui::MainWindow *ui;
private:
    void setupUI();
    void setupModel();

    QListView *listView;
    QStringListModel *model;
    SimpleDelegate *delegate;



};
#endif // MAINWINDOW_H

.cpp

#include "mainwindow.h"
#include "./ui_mainwindow.h"

#include <QListView>
#include <QStringListModel>
#include <QStyledItemDelegate>
#include <QPainter>
#include <QVBoxLayout>
#include <QWidget>
class SimpleDelegate : public QStyledItemDelegate
{
public:
    explicit SimpleDelegate(QObject *parent = nullptr) : QStyledItemDelegate(parent) {}

    void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const override
    {
        // 绘制背景
        if (option.state & QStyle::State_Selected) {
            painter->fillRect(option.rect, QColor(173, 216, 230)); // 浅蓝色选中背景
        } else {
            painter->fillRect(option.rect, Qt::white); // 白色默认背景
        }

        // 绘制文本
        QString text = index.data(Qt::DisplayRole).toString();
        painter->setPen(Qt::black);
        painter->drawText(option.rect, Qt::AlignCenter, text);

        // 绘制边框
        painter->setPen(Qt::lightGray);
        painter->drawRect(option.rect.adjusted(0, 0, -1, -1));
    }

    QSize sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const override
    {
        // 设置项目大小
        QSize size = QStyledItemDelegate::sizeHint(option, index);
        size.setHeight(40); // 固定高度
        return size;
    }
};

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
    , listView(new QListView(this))
    , model(new QStringListModel(this))
    , delegate(new SimpleDelegate(this))
{
     ui->setupUi(this);
    setupUI();
    setupModel();

    setWindowTitle("QT6 MVD 简单示例");
    resize(300, 300);

}

void MainWindow::setupUI()
{
    // 创建中央部件
    QWidget *centralWidget = new QWidget(this);
    setCentralWidget(centralWidget);

    // 设置列表视图属性
    listView->setModel(model);
    listView->setItemDelegate(delegate);
    listView->setAlternatingRowColors(true); // 交替行颜色

    // 创建布局
    QVBoxLayout *layout = new QVBoxLayout(centralWidget);
    layout->addWidget(listView);
}

void MainWindow::setupModel()
{
    // 创建示例数据
    QStringList data;
    data << "任务1: 学习 QT6"
         << "任务2: 理解 MVD 模式"
         << "任务3: 编写示例代码"
         << "任务4: 测试应用程序"
         << "任务5: 优化界面效果";

    // 设置模型数据
    model->setStringList(data);
}
MainWindow::~MainWindow()
{
    delete ui;
}

结果:

功能演示:

我可一只修改delegate中的 painter->fillRect(option.rect, Qt::white); // 白色默认背景将背景变为red;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值