头部视图拉伸放大效果实现原理解析

stretchableView.gif

在很多APP中大家应该都见过一些类似个人主页的页面,下边是tableview列表,上边的头部视图可以拉伸放大.

在一番研究实现了这个拉伸效果后,顺便把这个功能进行了封装,使用时只需两行代码

1.初始化调用

stretchableView = LPStretchableHeaderView(stretchableView: bgImageView)

2.代理方法实现:

func scrollViewDidScroll(_ scrollView: UIScrollView) {
    stretchableView.scrollViewDidScroll(scrollView)
}

github代码demo


下边是实现原理的解析,大神请绕路~~~

实现原理解析

一:设置头部视图
创建imageView,并添加到控制器的view上

由于在往下拖动列表时,头部视图的y值是没有跟着下移的,所以肯定不能让它作为tableview的tableHeaderView,只能把上边的图片视图添加到Controller的view上

二:设置tableview
  • 创建一个跟头部视图同样大小的空视图headerView作为tableview的tableHeaderView,来填充头部图片视图区域
  • 把tableview的背景颜色设置为clearColor,这样就可以看到下面的头部图片了

这样一来,我们的列表视图的实际大小就占据了整个屏幕,并且不影响看到下面的头部图片,而且在头部图片区域拖动的时候(实际拖动的是列表的tableHeaderView)也可以触发列表的滚动事件,同时上滑的时候列表的顶部滚动区域也达到了导航栏位置,一石好几鸟啊~😎

三:头部视图添加子控件

一般在头部视图的图片上方,还会显示昵称、头像等信息,我们需要把这些子控件添加到刚才创建的空视图headerView

注意:不要添加到头部视图的imageView中!

因为稍后我们在在拉伸列表时,会改变imageView的frame,但是并没有改变其内的子控件的frame,所以子控件位置会发生错乱。

现在控件布局如下:

override func viewDidLoad() {
    super.viewDidLoad()
    
    // 头部图片视图
    bgImageView = UIImageView(frame: CGRect(x: 0, y: 0, width: SCREEN_WIDTH, height: SCREEN_WIDTH * imageRatio))
    bgImageView.image = UIImage(named: "123")
    view.addSubview(bgImageView)
    
    // 列表
    tableView = UITableView(frame: CGRect(x: 0, y: 0, width: SCREEN_WIDTH, height: SCREEN_HEIGHT))
    tableView.register(UITableViewCell.self, forCellReuseIdentifier: "cell")
    tableView.dataSource = self
    tableView.delegate = self
    tableView.showsVerticalScrollIndicator = false
    tableView.backgroundColor = UIColor.clear // 注意要清除列表的背景颜色
    view.addSubview(tableView)
    
    // 创建一个空白view来进行填充tableHeaderView
    let headerView = UIView(frame: bgImageView.bounds)
    tableView.tableHeaderView = headerView
    
    // 添加label子控件
    let nameLabel = UILabel(frame: CGRect(x: 0, y: 150, width: bgImageView.width, height: 40))
    nameLabel.text = "哈哈哈😆"
    nameLabel.textAlignment = .center
    nameLabel.textColor = UIColor.white
    headerView.addSubview(nameLabel) // 注意要把子控件添加到headerView中
    
    // 导航栏
    makeNavView()
}
三:实现滚动拉伸放大效果
  • 下拉时通过tableview在y轴的偏移量来决定头部图片的高度拉伸多少
  • 通过图片拉伸的高度及图片原来的宽高比例计算出要拉伸的宽度
  • 通过图片拉伸后的宽度计算出x值应向左的偏移量

1)首先,我们会在头部视图初始化的时候记录它的原始frame,这个后边会用到

// 图片的原始frame
originFrame = bgImageView.frame

2)取出列表y值的偏移量

func scrollViewDidScroll(_ scrollView: UIScrollView) {
    let yOffset = scrollView.contentOffset.y
}

3)图片拉伸后的高度

// 下拉时yOffset值是负数,所以需要减
frame.size.height = originFrame.size.height - yOffset

4)图片拉伸后的宽度

// 通过图片的宽高比imageRatio及拉伸后的高度等比计算新宽度
frame.size.width = frame.size.height / imageRatio

5)x值的位置重新计算

// 图片宽高同时变大后,图片会整体向右偏移,所以需要重新计算x值
frame.origin.x = originFrame.origin.x - (frame.size.width - originFrame.size.width) * 0.5

当列表上滑时,移动头部图片跟着向上移动

var frame = originFrame
frame.origin.y = originFrame.origin.y - yOffset
bgImageView.frame = frame

最终,处理代码为:

func scrollViewDidScroll(_ scrollView: UIScrollView) {
    
    let yOffset = scrollView.contentOffset.y
    
    // 头部图片拉伸设置
    if yOffset > 0 { // 上滑
        var frame = originFrame
        frame.origin.y = originFrame.origin.y - yOffset
        bgImageView.frame = frame
        
    } else { // 下拉
        var frame = originFrame
        frame.size.height = originFrame.size.height - yOffset
        frame.size.width = frame.size.height / imageRatio
        frame.origin.x = originFrame.origin.x - (frame.size.width - originFrame.size.width) * 0.5
        bgImageView.frame = frame
    }
}



封装

通过以上实现可以看出,所有的操作都是通过拿到scrollViewDidScroll回调方法中列表y轴的偏移量,然后对头部视图bgImageView的frame进行更改实现的。

所以其实没啥好封装的,如果非得封装的话,那就只需要把bgImageView控件和对它frame更改的操作拿出去就ok了。

于是我建了一个工具类LPStretchableHeaderView,实现如下:

LPStretchableHeaderView.swift文件

import UIKit

public class LPStretchableHeaderView: NSObject {

    private var stretchView = UIView()
    private var imageRatio: CGFloat
    private var originFrame = CGRect()
    
    public init(stretchableView: UIView) {
        
        stretchView = stretchableView
        originFrame = stretchableView.frame
        imageRatio = stretchableView.bounds.height / stretchableView.bounds.width
    }
    
    public func scrollViewDidScroll(_ scrollView: UIScrollView) {
        
        let yOffset = scrollView.contentOffset.y
        if yOffset > 0 { // 往上移动
            var frame = originFrame
            frame.origin.y = originFrame.origin.y - yOffset
            stretchView.frame = frame
        } else { // 往下移动
            var frame = originFrame
            frame.size.height = originFrame.size.height - yOffset
            frame.size.width = frame.size.height / imageRatio
            frame.origin.x = originFrame.origin.x - (frame.size.width - originFrame.size.width) * 0.5
            stretchView.frame = frame
        }
    }
}

这样,以后遇到有这种需求的页面,两行代码就搞定了~


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

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,082评论 4 62
  • 亲爱的陆钰你好: 今天是挺有趣的一天,早晨因为老公贴心地主动送东东去上英语课,让你有时间去apple专卖店修理破碎...
    常拓阅读 203评论 3 7
  • 1、SMART原则:适用于目标设定、任务委派 Specific 具体明确的 Measurable 可衡量的 Ach...
    行者潘阅读 2,040评论 0 3
  • 如果东海水全部下成雨, 落到火焰山, 是山火被浇灭, 还是海水被烘干? 每隔几年, 会有一批老实忠厚的字眼, 从教...
    青舟青舟阅读 111评论 0 4
  • 实际上编程并不像大家想象中的那样you多深奥,you多难懂。似乎只you站在业界最前沿的一小撮人士才能办到。其实只...
    NEYO_47a8阅读 502评论 0 0