剖析支付宝首页TableView的结构

引言:今天这篇文章属于分析型,我将一步步带你走进我的思路,跟我一起头脑风暴,欢迎加入。
内容介绍:支付宝首页结构看起来很简单,无非就是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在TableView上面,因此只需要让Header的y等于0就可以了。(在scrollVIew的滑动就会触发的代理方法里设置这些scrollViewDidScroll
// 上面的第二步设置头的代码
tableView.tableHeaderView = UIView(frame: CGRect(x: 0, y: 0, width: 0, height: HeaderViewHeight))

结束:至此,我们根据上面的思路就可以做出支付宝首页的结构了,剩下的无非就是Header的按钮s,以及自定义的TableViewCell,架子就是这个样子。

Demo下载链接

如果觉得写得还行,点个Star呗
同时也欢迎评论中指出本文存在的bug,或者疑问,互相促进!
作者邮箱:pangshishan@aliyun.com
github地址:https://github.com/Pangshishan
qq/微信: 704158807

代码:

import UIKit
import MJRefresh

func ScreenWidth() -> CGFloat {
    return UIScreen.main.bounds.width
}
func ScreenHeight() -> CGFloat {
    return UIScreen.main.bounds.height
}

let HeaderViewHeight: CGFloat = 150
let HeaderBGColor = UIColor.red
let TableViewCellID = "TableViewCellID"
let ProgressControlWidth: CGFloat = 6

class ViewController: UIViewController {

    fileprivate lazy var headerView = UIView()
    fileprivate var tableView: UITableView!
    fileprivate var cellCount: Int = 30
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        setupUI()
    }
}

// MARK:- 设置UI
extension ViewController {
    fileprivate func setupUI() {
        setupTableView()
        setupHeaderView()
        setupScrollProgressControl()
        setupRefresh()
    }
    fileprivate func setupHeaderView() {
        headerView.frame = CGRect(x: 0, y: 0, width: ScreenWidth(), height: HeaderViewHeight)
        headerView.backgroundColor = HeaderBGColor
        tableView.addSubview(headerView)
        
    }
    fileprivate func setupTableView() {
        let tRect = CGRect(x: 0, y: 0, width: ScreenWidth(), height: ScreenHeight())
        tableView = UITableView(frame: tRect, style: .grouped)
        view.addSubview(tableView)
        tableView.delegate = self
        tableView.dataSource = self
        tableView.register(UITableViewCell.self, forCellReuseIdentifier: TableViewCellID)
        tableView.tableHeaderView = UIView(frame: CGRect(x: 0, y: 0, width: 0, height: HeaderViewHeight))
        tableView.showsVerticalScrollIndicator = false
    }
    fileprivate func setupScrollProgressControl() {
        let pFrame = CGRect(x: ScreenWidth() - 0 - ProgressControlWidth, y: HeaderViewHeight, width: ProgressControlWidth, height: ScreenHeight() - HeaderViewHeight)
        let progressC = ScrollProgressControl(frame: pFrame, scrollView: tableView)
        view.addSubview(progressC)
    }
    
    fileprivate func adjustHeaderView() {
        let offSetY = tableView.contentOffset.y
        if offSetY <= 0 {
            headerView.frame.origin.y = offSetY
        } else {
            headerView.frame.origin.y = 0
        }
       
    }
    fileprivate func setupRefresh() {
        tableView.mj_header = MJRefreshNormalHeader.init(refreshingBlock: { [weak self] in
            self?.perform(#selector(self?.setupHeaderData), with: nil, afterDelay: 1.5)
        })
        tableView.mj_header.ignoredScrollViewContentInsetTop = -HeaderViewHeight
        
        tableView.mj_footer = MJRefreshAutoNormalFooter(refreshingBlock: { [weak self] in
            self?.perform(#selector(self?.setupFooterData), with: nil, afterDelay: 0.5)
        })
    }
}

// tableView代理方法
extension ViewController: UITableViewDelegate, UITableViewDataSource {
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return cellCount
    }
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: TableViewCellID, for: indexPath)
        cell.textLabel?.text = "\(indexPath.row)"
        return cell
    }
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        tableView.deselectRow(at: indexPath, animated: true)
    }
    func tableView(_ tableView: UITableView, heightForHeaderInSection section: Int) -> CGFloat {
        return 0.1
    }
    func tableView(_ tableView: UITableView, heightForFooterInSection section: Int) -> CGFloat {
        return 0.1
    }
}
// MARK:- 数据方法
extension ViewController {
    @objc fileprivate func setupHeaderData() {
        cellCount = 30
        tableView.reloadData()
        self.tableView.mj_header.endRefreshing()
    }
    @objc fileprivate func setupFooterData() {
        cellCount += 30
        tableView.reloadData()
        self.tableView.mj_footer.endRefreshing()
    }
}

// scrollView代理方法
extension ViewController: UIScrollViewDelegate {
    func scrollViewDidScroll(_ scrollView: UIScrollView) {
        if scrollView == tableView {
            adjustHeaderView()
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352

推荐阅读更多精彩内容