News Digest(雅虎新闻)模仿秀第四弹

最近得空做了一个小的新闻类APP,基本上都是照着News Digest的模子刻出来的,之所以这个为参考,是因为觉得News Digest这个APP做得真的很酷炫!


猛戳这里获取整个项目源代码
项目前端主要由swift编写,本地数据用CoreData,后端由Node.js编写,后台数据库用MongoDB。

News Digest(雅虎新闻)模仿秀第一弹
News Digest(雅虎新闻)模仿秀第二弹
News Digest(雅虎新闻)模仿秀第三弹

这期我们来说一下倒计时动画的实现,先上效果图:


倒计时效果图

这里要分两部分来讲,一部分是圆圈动画的填充,一部分是时间的快速滚动,位于项目CustomView/MenuView和CustomView/CountdownView

  1. 圆圈动画的执行
    init(timeType: TimeType) {
        super.init(frame: defaultFrame)
        type = timeType
        self.backgroundColor = UIColor.clearColor()
        
        // CAShaperLayer Setting
        circleShadowLayer.frame = self.bounds
        circleShadowLayer.fillColor = UIColor.clearColor().CGColor
        circleShadowLayer.lineWidth = 2.0
        circleShadowLayer.strokeEnd = 1.0
        circleShadowLayer.strokeColor = UIColor.RGBColor(255, green: 255, blue: 255, alpha: 0.2).CGColor
        
        circleLayer.frame = self.bounds
        circleLayer.fillColor = UIColor.clearColor().CGColor
        circleLayer.lineWidth = 2.0
        circleLayer.strokeEnd = 0
        circleLayer.strokeColor = UIColor.RGBColor(0, green: 121, blue: 166, alpha: 1).CGColor
        
        self.layer.addSublayer(circleShadowLayer)
        self.layer.addSublayer(circleLayer)
    }

这里的circleShadowLayer,就是动图底层那一个灰色的圈,我们设置strokeEnd = 1.0(就是说完全填充,形成一个闭环),这个strokeEnd是从0.0-1.0范围

    override func layoutSubviews() {
        super.layoutSubviews()
        
        defaultCircleRadius = (WIDTH - 100) < (self.bounds.height - 32) ? (WIDTH - 100)/2.0 : defaultCircleRadius
        circleShadowLayer.frame = self.bounds
        circleShadowLayer.path = circlePath().CGPath
        circleLayer.frame = self.bounds
        circleLayer.path = circlePath().CGPath
    }
    
    //MARK: - CirclePath
    func circlePath() -> UIBezierPath {
        return UIBezierPath.init(ovalInRect: circleFrame())
    }
    
    func circleFrame() -> CGRect {
        var circleFrame = CGRect(x: 0, y: 0, width: 2*defaultCircleRadius, height: 2*defaultCircleRadius)
        circleFrame.origin.x = CGRectGetMidX(circleShadowLayer.bounds) - CGRectGetMidX(circleFrame)
        circleFrame.origin.y = CGRectGetMidY(circleShadowLayer.bounds) - CGRectGetMidY(circleFrame)
        return circleFrame
    }

circleFrame方法是设置圆圈的frame,让它形成位于圆心的,半径为defaultCircleRadius的圆,然后用UIBezierPath绘制路径,每次调用layoutSubviews方法把路径赋值给circleLayer
layoutSubviews调用时机:

  - 直接调用setLayoutSubviews。
  - addSubview的时候。
  - 当view的frame发生改变的时候。
  - 滑动UIScrollView的时候。
  - 旋转Screen会触发父UIView上的layoutSubviews事件。
  - 改变一个UIView大小的时候也会触发父UIView上的layoutSubviews事件。

期初我打算用UIView.animateWithDuration来赋值circleLayer的strokeEnd参数,然后发现无论我设置duration是多少,动画执行时间都是一样的,后来查了一下资料才发现UIView.animateWithDuration只能动画改变这几个参数

        /*
         *   UIView.animateWithDuration(13.2, animations: {
         *       self.circleLayer.strokeEnd = CGFloat(interval/43200)
         *   })
         *   这里执行方式不可以使用UIView.animateWithDuration,只能用CAAnimations
         *   UIView动画执行只能改变一下几个参数
         *
         *   The following properties of the UIView class are animatable:
         *   @property frame
         *   @property bounds
         *   @property center
         *   @property transform
         *   @property alpha
         *   @property backgroundColor
         *   @property contentStretch
         */

所以动画执行strokeEnd的改变只能使用CAAnimations,只需要把strokeEnd的数值赋值给toValue,然后设置执行时间即可

    func animationExecute() {
       circleLayer.addAnimation(self.circleAnimationImplement(1.6, delay: 0.3, toValue: interval!/43200.0), forKey: nil)
    }
    
    // Animation Implement
    private func circleAnimationImplement(duration: NSTimeInterval, delay: Double, toValue: Double) -> CABasicAnimation {
        let animation = CABasicAnimation.init(keyPath: "strokeEnd")
        animation.duration = duration < 0.8 ? 0.8 : duration
        animation.beginTime = CACurrentMediaTime() + delay
        animation.fromValue = 0.0
        animation.toValue = toValue
        animation.removedOnCompletion = false // 这里如果不是false fillMode属性不起作用
        animation.fillMode = kCAFillModeForwards; // 保留动画后的样子
        // 设置Delegate
        animation.delegate = self
        return animation
    }
  1. 时间的快速滚动效果
    其实中间就是一个UILabel,然后动态修改它的attributedText即可
    这个数字快速滚动效果,我是模仿下面这位大大的代码写的
    滚动的数字:FlickerNumber
    如果需要详细了解,戳上面这个链接就可以详细了解.
    总的来说,大概实现思路就是:
  • 设定起始数字/终止数字/执行时间等
  • 设定数字滚动的频率,比如说1/30等等
  • 然后根据终止数字*频率/执行时间,就获得每次数字变化的值
  • 最后一个NSTimer以频率来执行改变UILabel的值
  1. 滚动圈圈下面那个日期选择
    具体就不细说了,就一个UICollectionView

今天就到这里了。

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

推荐阅读更多精彩内容