翻译篇:实现Twitter个人详情动态效果

原文由ariok发表,地址是implementing-the-twitter-ios-app-ui

效果图如下:

效果图

原来作者的代码会存在一个概现Bug:当快速下拉时,个人头像并不会立刻显示在HeaderView上方,我已经向作者提交了Pull requests

编译过程中会发生错误,因为swift更新了,所以需要自己解决下错误。

翻译的比较差,因为不做阅读理解好多年了。。。重点在效果。。。

结构描述

在编码之前,我将对UI的结构做一个简单的介绍。

打开Main.storyboard文件。在唯一的一个控制器view中,你可以发现两个主要的对象。第一个是显示Header的视图,第二个是一个包含个人头像(我们叫它Avatar)和其他与账号相关,比如用户名、follow按钮的ScrollView。Sizer控件只是用来确认ScrollView内容是否能进行垂直滚动。

就像你所看到的,这个结构非常简单。需要的注意的是,我将Header放在ScrollView外面,而不是把它和其他ScrollView子控件放在一起。因为这样做可以让这个结构具备更好的扩展性。

开始编码

如果你仔细看完动画,你会注意到你可以管理两个不同的动作:

  1. 用户下拉(当ScrollView内容已经在屏幕的顶部时)

  2. 用户上/下滚动

这个动作可以分解成四步:

2.1) 向上滚动,Header控件缩小直到它的尺寸和导航栏默认尺寸相等,然后这个Header控件就会粘在屏幕的上方

2.2) 向上滚动, Avatar(头像)逐渐变小

2.3) 当Header控件和ScrollView的子控件重叠时,Avatar(头像)在Header控件底部

2.4) 当用户名Label的顶部和Header控件底部重叠时,一个新的白色Label将会从Header控件的中底部显示。并且Header控件的图片变模糊。

打开ViewController.swift,让我们一步一步地实现这些功能

设置控制器

第一件需要去做的事是获取ScrollView的offset信息。通过实现UIScrollViewDelegate协议的scrollViewDidScroll方法,我们可以很容易地做到这一点。

一种最简单的展示一个view上变化的方式是使用CoreAnimation三维变换,并且设置新值给layer.transform属性。

这个关于CoreAnimation的教程可能会让它变得简便:

http://www.thinkandbuild.it/playing-around-with-core-graphics-core-animation-and-touch-events-part-1/.

以下是scrollViewDidScroll方法的第一部分:

let offset = scrollView.contentOffset.y
var avatarTransform = CATransform3DIdentity
var headerTransform = CATransform3DIdentity

我们可以在这个方法里面获取当前的竖直偏移,并且初始化两个将要在方法后面设置的转换信息。

下拉

让我们对下拉动作进行处理:

if offset < 0 {
    
    let headerScaleFactor:CGFloat = -(offset) / header.bounds.height
    let headerSizevariation = ((header.bounds.height * (1.0 + headerScaleFactor)) - header.bounds.height)/2.0
    headerTransform = CATransform3DTranslate(headerTransform, 0, headerSizevariation, 0)
    headerTransform = CATransform3DScale(headerTransform, 1.0 + headerScaleFactor, 1.0 + headerScaleFactor, 0)
    
    header.layer.transform = headerTransform
}

首先,我们检查偏移量是否为负数,即ScrollView是否已出现弹性区域。

剩下的代码是一些简单的数学运算。

这个Header控件需要放大来保持它的上边缘和屏幕顶部相对固定,并且这个图片是从底部开始放大的。

总的来说,这个变换主要由缩放,然后转化view的尺寸变化为到顶部的距离构成。事实上,你可以朝屏幕顶端移动ImageView图层的中点并且进行缩放来相同的效果。

使用一个属性来对头部缩放比例进行计算。我们希望Header控件参照偏移量进行适当的缩放。换种说法:当偏移量为Header视图控件的两倍时,头部缩放比例应该设置为2.0。

我们需要处理的第二个动作是上下滚动。让我们看看如何一步一步地完成主要视图的变换。

Header(第一阶段)

当前的偏移量应该大于0。Header控件应该根据以下的偏移量来进行竖直移动,直到它到达指定高度(我们下面将会对Header模糊进行讲解)。

headerTransform = CATransform3DTranslate(headerTransform, 0, max(-offset_HeaderStop, -offset), 0)

这段代码真的是很简单。我们只需要设置Header控件偏移一个最小值,Header控件将会在offset_HeaderStop这个点停止移动。

因为我比较懒,所以我写死了一些数值,比如offset_HeaderStop。我们可以通过更加优雅的方式,比如计算UI控件的位置来实现相同的效果。或许在下一次我会试试。

AVATAR(头像)

这个头像(图片)以和下拉相同的逻辑进行缩放,只是在这种情况下,图片是和底部贴合而不是顶部。这段代码和上面比较相似,除了减小缩放的比例为1.4。

let avatarScaleFactor = (min(offset_HeaderStop, offset)) / avatarImage.bounds.height / 1.4 // Slow down the animation
let avatarSizeVariation = ((avatarImage.bounds.height * (1.0 + avatarScaleFactor)) - avatarImage.bounds.height) / 2.0
avatarTransform = CATransform3DTranslate(avatarTransform, 0, avatarSizeVariation, 0)
avatarTransform = CATransform3DScale(avatarTransform, 1.0 - avatarScaleFactor, 1.0 - avatarScaleFactor, 0)

就像你看到的,当Header控件停止变化时,我们通过min函数来停止对个人头像的缩放(offset_HeaderStop)。

此时,我们根据当前的偏移量来设置最顶层的图层。除非偏移量大于等于offset_HeaderStop,否则顶部图层始终是个人头像。当偏移量大于offset_HeaderStop,这个图层就变成了Header控件。

if offset <= offset_HeaderStop {
    
    if avatarImage.layer.zPosition < header.layer.zPosition{
        header.layer.zPosition = 0
    }
    
}else {
    if avatarImage.layer.zPosition >= header.layer.zPosition{
        header.layer.zPosition = 2
    }
}

白色Label

以下是白色Label执行动画的代码:

let labelTransform = CATransform3DMakeTranslation(0, max(-distance_W_LabelHeader, offset_B_LabelHeader - offset), 0)
headerLabel.layer.transform = labelTransform

这里介绍两个新的变量:当偏移量等于offset_B_LabelHeader时,这个黑色的用户名Label刚好到达Header视图的底部。

distance_W_LabelHeader是Header控件的底部和Header中的白色Label中点的距离。


这个转换通过以下逻辑进行计算:黑色Label一旦喝Header控件相交,白色Label就立即显示,并且白色Label到达Header控件的中点时停止。所以使用以下代码来创建Y的偏移:

max(-distance_W_LabelHeader, offset_B_LabelHeader - offset)

模糊

最后一个效果是模糊Header控件。为了得到合适的解决方案,我使用了三个不同的库...我还尝试创建自己的OpenGL ES,但是实时更新模糊效果总是非常迟缓。

我了解到我可以只对模糊进行一次计算,让模糊和非模糊的图片进行重叠,并且改变透明度值。我很确定,这就是Twitter采用的方法。

在viewDidAppear中我们计算模糊的Header并且通过设置透明度为0来进行隐藏。

headerBlurImageView = UIImageView(frame: header.bounds)
headerBlurImageView?.image = UIImage(named: "header_bg")?.blurredImageWithRadius(10, iterations: 20, tintColor: UIColor.clearColor())
headerBlurImageView?.contentMode = UIViewContentMode.ScaleAspectFill
headerBlurImageView?.alpha = 0.0
header.insertSubview(headerBlurImageView, belowSubview: headerLabel)

模糊的view可以使用FXBlurView得到。

在scrollViewDidScroll方法中,我们只需要根据偏移量来更新透明度就行了。

headerBlurImageView?.alpha = min (1.0, (offset - offset_B_LabelHeader)/distance_W_LabelHeader)

以上计算逻辑主要为:透明度最大值必须为1,模糊效果必须在黑色Label到达Header控件时出现,在白色Label停止后停止加深模糊。

源码地址

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

推荐阅读更多精彩内容