剖析支付宝首页TableView的结构

本文详细分析了支付宝首页TableView的结构,重点探讨了下拉刷新时的头部进度条和Header的移动机制。通过实验推断,得出进度条是自定义实现的结论,并提供了封装进度条的方法。同时,解释了如何让头部刷新控件出现在合适位置的技巧,为复现类似效果提供了思路。
摘要由CSDN通过智能技术生成

引言:今天这篇万丈属于分析型,我将一步步带你走进我的思路,跟我一起头脑风暴,欢迎加入。
内容介绍:支付宝首页结构看起来很简单,无非就是TableView+Header;但是当我们仔细分析的时候,发现有一样东西用原生的TableView是无法实现。那就是TableView左侧的阅读进度条。接下来我就为大家分析一下(我的做法只是推断,或许并非原生的做法,如果内部人员看到欢迎指正)

首先,我们先来看图,分析一下。我们分析的重点就两个地方,在图中已经标出来了,位置1位置2
支付宝首页截图

位置1:当下拉支付宝首页时,头部下拉刷新控件在位置1处出现。
位置2:当最开始让TableView向下走一点点的时候,TableView的进度条在位置2处出现。

通过以上两个位置,我最开始是认为,tableView只有下半屏幕的大小,如下图这么大
图片.png

我为什么这么以为呢,因为正常情况下,下拉刷新会出现在TableView的顶部,进度条最开始出现的地方也应该是TableView的顶部。那我们就来以这个思路往下走走看,假设他就这么大。

通过以上假设,我们可以提出一下几个问题:
1. TableView大小是半个屏幕,它是如何显示整个屏幕的东西的
2. 在TableView向上滑动的时候,为什么Header也会跟着往上走(此处Header指“扫一扫,付钱,收钱…;余额,花呗,充值中心…..”那部分)

下面我们来试着解决这些问题:
1. 可能Header和tableview都是加在了self.view 上了,当TableView向上滑动时,根据滑动的偏移量,让Header向上移动
2. 在第一步,我们不仅让Header向上移动,同时改变TableView的frame,让tableView慢慢变大,于是完美解决以上两个问题

但是,当我们认为问题解决的时候,新问题出现了,如果改变TableView的frame,让tableView慢慢变成self.view 的大小,右侧进度条也势必会向上移动,可是支付宝app并不是这样。因此较劲脑汁想的方法并不成立。

我新开了个工程,写了一个TableView,当我们下拉TableView的时候,右侧进度条并不会显示,但是下拉支付宝的时候,进度条依然存在;那么就说明:支付宝的进度条是假的,是自定义的

既然进度条是自定义的,那我们之前的推论(TableView大小是屏幕的一般)可能就不成立了,我们把TableView的进度条隐藏,换成我们自己封装的进度条。

此处我简单说一下怎么封装进度条用起来最简单吧,文章后面我会上代码的:1.自定义View,只需要把TableView传给这个View;2.用View监听TableView的contentOffSet属性;3.根据监听的结果显示进度条的长度以及运动;4.显示和隐藏的逻辑请看代码

进度条的问题解决完了,接下来我们解决头部刷新的问题

我们让TableView和屏幕一样大,那我们怎么让头部刷新控件出现在中间呢?答案很简单:
1. MJRefresh控件有一个属性ignoredScrollViewContentInsetTop,可以调整刷新控件的上下位置
2. 先添加TableView到self.view上,设置TableView的头部高度和Header一样高,然后添加Header到TableView上,正好盖在TableView的头部
3. 这一步我们解决tableView向上和向下滑动时,确定头部的位置;当TableView向下滑动时(contentOffset.y < 0),我们需要把Header向上移动,让Header相对于屏幕不动,就可以漏出挡住的头部刷新控件。当TableView向下滑动,因为Header在Tab

要在Qt中使用TableView显示自定义数据结构,你需要将该结构转换为Qt的模型(Model)。Qt的模型-视图(Model-View)框架提供了两种模型:标准模型和自定义模型。标准模型适用于表格和列表等简单的数据结构,而自定义模型则更加灵活,适用于显示自定义数据结构。 以下是一些步骤,可以帮助你在Qt中使用TableView显示自定义数据结构: 1. 创建自定义数据结构 首先,你需要创建一个自定义数据结构,例如一个包含姓名、年龄和性别的结构体。 ```c++ struct Person { QString name; int age; QString gender; }; ``` 2. 创建自定义模型 接下来,你需要创建一个自定义模型,该模型将用于将自定义数据结构转换为Qt模型。你可以创建一个继承自QAbstractTableModel的类,并重写以下方法: ```c++ class PersonModel : public QAbstractTableModel { public: PersonModel(QObject *parent = nullptr); int rowCount(const QModelIndex &parent = QModelIndex()) const override; int columnCount(const QModelIndex &parent = QModelIndex()) const override; QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override; QList<Person> persons; private: int _rowCount; int _columnCount; }; ``` 在上面的代码中,我们创建了一个名为PersonModel的自定义模型,并重写了rowCount、columnCount、data和headerData方法。 其中,rowCount和columnCount方法返回行数和列数,data方法返回单元格的数据,headerData方法返回表头的数据。 3. 实现自定义模型的方法 接下来,你需要实现自定义模型的方法。在这里,你需要将自定义数据结构转换为Qt模型的数据。在PersonModel类中,我们使用了一个QList来存储Person结构体的数据,因此我们需要在重写的方法中将该数据转换为Qt模型的数据。 ```c++ int PersonModel::rowCount(const QModelIndex &parent) const { Q_UNUSED(parent); return persons.count(); } int PersonModel::columnCount(const QModelIndex &parent) const { Q_UNUSED(parent); return 3; } QVariant PersonModel::data(const QModelIndex &index, int role) const { if (!index.isValid()) return QVariant(); if (index.row() >= persons.size() || index.row() < 0) return QVariant(); if (role == Qt::DisplayRole || role == Qt::EditRole) { const auto &person = persons.at(index.row()); if (index.column() == 0) return person.name; else if (index.column() == 1) return person.age; else if (index.column() == 2) return person.gender; } return QVariant(); } QVariant PersonModel::headerData(int section, Qt::Orientation orientation, int role) const { if (role != Qt::DisplayRole) return QVariant(); if (orientation == Qt::Horizontal) { switch (section) { case 0: return tr("Name"); case 1: return tr("Age"); case 2: return tr("Gender"); default: return QVariant(); } } return QVariant(); } ``` 在上面的代码中,我们重写了rowCount、columnCount、data和headerData方法,并将Person结构体的数据转换为Qt模型的数据。 4. 使用TableView显示自定义数据结构 最后,你可以使用TableView来显示自定义数据结构。在MainWindow类中,你可以创建一个TableView,并将它设置为PersonModel的视图。 ```c++ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) , ui(new Ui::MainWindow) { ui->setupUi(this); // create person model PersonModel *personModel = new PersonModel(this); personModel->persons.append({ "Tom", 25, "Male" }); personModel->persons.append({ "Alice", 30, "Female" }); personModel->persons.append({ "John", 40, "Male" }); // create table view QTableView *tableView = new QTableView(this); tableView->setModel(personModel); // set table view properties tableView->setSelectionMode(QAbstractItemView::SingleSelection); tableView->setSelectionBehavior(QAbstractItemView::SelectRows); tableView->setEditTriggers(QAbstractItemView::NoEditTriggers); // set central widget setCentralWidget(tableView); } ``` 在上面的代码中,我们创建了一个PersonModel,并将它设置为TableView的模型。然后,我们设置了TableView的一些属性,并将它设置为MainWindow的中央窗口部件。 这样,你就可以使用TableView来显示自定义数据结构了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值