仿UC头条滑动效果

仿UC头条滑动效果

一、UC头条的滑动效果

UC头条的滑动效果有两个地方:

  1. 在主页面的滑动过程中,左右两个页面中的子页面与滑动方向相反而形成交错的效果

  2. 页签标题下的指示视图在滑动过程中进行了拉升动画

    1. 在页面滑到一半的时候指示视图拉升到下一个页签的终止位置,而保持起始位置不变

    2. 在页面进行下一半滑动的时候指示视图进行收缩到下一个页签的起始位置,而保持终止位置不变

二、分析

对于滑动动画来说,一般都是有其周期性的。对于这里的两个动画都有其周期性。所以我们可以将其动画转换为一个固定的动画。这个动画由一个固定的输入,对应于一个固定的输出:y=f(x)。其实一开始我们会有点没头绪,但是耐心下来分析的时候,其实这两个动画都可以归结为两个线性函数。

三、实现方式

页面视图

1、移动效果

这个页面的滑动只是简单的交错滑动效果:offsetXStep(滑动的偏移量)->centerXStep(中心的偏移量),我们这里就是讲滑动页面每次的滑动偏移的量转换为我们子页面需要偏移的值。其关键代码为

CGFloat offset = scrollView.contentOffset.x - contentOffsetBefore;
contentOffsetBefore = scrollView.contentOffset.x;
CGFloat scale = (self.bounds.size.width/2) / scrollView.bounds.size.width;
UIView *firstV = self.elementsV[index];
firstV.center = CGPointMake(firstV.center.x + offset * scale, firstV.center.y);
if (index + 1 <= self.elementsV.count - 1) {
    UIView *nextV = self.elementsV[index + 1];
    nextV.center = CGPointMake(nextV.center.x + offset * scale, nextV.center.y);
}

因为我们每次滑动的时候只会有两个页面同时显示出来,所以我们这里需要进行移动操作的视图只需要当前,以前下一个页面

2、初始位置

为了实现这种偏移效果,视图的原先位置必须有所偏移,如果是当前的,则不偏移,如果是当前左边的则向右偏移0.5个视图长度,如果是当前右边的则向左偏移0.5个视图长度。这也是这段代码的意义所在

for (int i = 0; i < self.elementsV.count; i++) {
     UIView *v = self.elementsV[i];
    if (i == currentIndex) {
        v.center = (CGPoint){v.bounds.size.width / 2, v.bounds.size.height / 2};
    } else if (i < currentIndex){
        v.center = (CGPoint){v.bounds.size.width, v.bounds.size.height / 2};
    } else if (i > currentIndex) {
        v.center = (CGPoint){0, v.bounds.size.height / 2};
    }
}
3、位置恢复

我们在计算位移的过程中,系统给我们的反馈,或者是自己的浮点型的运算总是会丢掉一些数值。如果我们不在每次滑动完毕的时候或者特定的时候进行位置的复原,那么在我们快速、不断的滑动过程中会出现视图位置的明显偏移,典型就是原本视图正好全屏于手机,结果却出现了偏移的情况。为了结果这个问题,我们可以在特定的时机调用#2方法来复原。例子中采用了如下的时机点

if ([self isReachedPagedOffset:scrollView.contentOffset.x]) {
    for (int i = 0; i < self.elementsV.count; i++) {
        UIView *showV = self.elementsV[i];
        if (index == i) {
            showV.center = [self.elementCenter[0] CGPointValue];
        }
    }
}

- (BOOL)isReachedPagedOffset:(CGFloat)offset {
    for (int i = 0; i < self.elementsV.count; i++) {
        if (offset == i * self.scrollV.bounds.size.width) {
            _currentIndex = i;
            return YES;
        }
    }
    
    return NO;
}

当偏移量为正好为当前视图的一个page的时候进行位置的复原。这样,不管我们怎么样进行拖动、切换,我们总能进行复原,这样就不会导致不可逆的偏差

标题视图

1、移动效果

因为我们可以左滑及右滑,我们将其分4段进行了处理,起周期性可以归结为[-1, 1]。一开始的时候我的周期性设置为[0, 1],当时的情况是左滑动没有问题,但是右滑出现了问题,后来分析,我自己的周期性取的有问题才改为[-1, 1]。关键代码如下

// 2: 当前段中的处理
CGRect newFrame = CGRectZero;

// 分两段处理,小于0.5的拉升,大于等于0.5的收缩
if (0 < progress && progress < 0.5) {
    // 2.1: 拉升
    CGFloat strechLength = progress / 0.5 * titleLenght;
    newFrame = (CGRect){normaFrame.origin.x, normaFrame.origin.y, normaFrame.size.width + strechLength, normaFrame.size.height};
} else if(progress >= 0.5 && progress != 1) {
    // 2.2: 收缩
    CGFloat shrinkLength = (progress - 0.5) / 0.5 * titleLenght;
    newFrame = (CGRect){normaFrame.origin.x + shrinkLength, normaFrame.origin.y, normaFrame.size.width + titleLenght - shrinkLength, normaFrame.size.height};
} else if (-0.5 < progress && progress < 0) {
    // 2.3: 拉升
    CGFloat strechLength = -progress / 0.5 * titleLenght;
    newFrame = (CGRect){normaFrame.origin.x - strechLength, normaFrame.origin.y, normaFrame.size.width + strechLength, normaFrame.size.height};
} else if (-1.0 < progress && progress <= 0.5) {
    // 2.4: 收缩
    CGFloat shrinkLength = -(progress + 0.5) / 0.5 * titleLenght;
    newFrame = (CGRect){normaFrame.origin.x -  ((self.currentIndex == 0) ? 0 : titleLenght), normaFrame.origin.y, normaFrame.size.width  + titleLenght - shrinkLength, normaFrame.size.height};
}

// 更新当前的位置
self.slideV.frame = newFrame;

这里同样有进行位置的复原处理,需要进行位置复原的原因上面已经说了。可以知道,位置复原的处理在滑动动画中是不可缺少的。

视觉效果

完成后的视觉如下

uclike.gif

补充

在词场中的登录页面也涉及了大量的滑动动画,那边的动画关注的不是每次的偏移量,而是当前的contentOffsetX来进行计算。一般的滑动动画的处理方式

  1. 元素的变更根据每次的偏移量来进行变化
  2. 元素的变更根据每次的contentOffsetX来直接对应
  3. 记得在滑动过程中在特定的时机进行元素的位置复原

代码下载

仿UC头条滑动动画代码下载

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,263评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,978评论 4 60
  • 今天听书《自控力》,精华讲解: 这一周,试着观察你意志力挑战成功和失败的时候,你是怎么对自己和他人解释的。 •当你...
    Rainbow小雪阅读 207评论 4 0
  • 人们总是有迷惘的吗?究竟怎么才能做到抵住其他的诱惑,全力以赴地去对准一个目标,打击下去。 毛爷爷说,伤敌人十指,不...
    Jimuboy阅读 1,254评论 0 0