iOS 弹幕实现分析及原理

弹幕需求

  • 弹幕在屏幕上以一定的轨道移动
  • 根据弹幕长度定移动速度
  • 弹幕文字支持自定义(字体、格式、颜色等)
  • 弹幕支持暂停及恢复

实现主要要解决的问题

  • 弹幕如何绘制?
  • 弹幕如何控制移动速度?
  • 弹幕如何检测是否碰撞?
  • 弹幕如何暂停移动及恢复移动?

注:“弹幕碰撞”指弹幕移动过程中前一条展示与后一条展示不出现重叠。

根据已知的几个问题,依次提出解决方案。

实现弹幕

弹幕如何绘制

Android 平台DanmakuFlameMaster库中,通过在 View 层的一帧帧的绘制来显示弹幕。对于每一条来说,如果要实现弹幕的流畅性,需要保证每秒绘制的帧数处在一个较高的数字。如果出现大量的弹幕,同时绘制,对设备的性能要求也会很高。所以自定义绘制帧的方式,不一定适用于所有的设备。

除了自定义绘制帧的方式展示弹幕,还可以通过采用系统动画的方式来实现弹幕文本的移动。例如 UIView 动画、Facebook/pop 动画。以系统 UIView 动画为例:

[UIView animateWithDuration:duration delay:0 options:UIViewAnimationOptionCurveLinear animations:^{
        // bulletLabel 为弹幕内容 label
        bulletLabel.frame = CGRectMake(-bulletLabel.width, bulletLabel.y, bulletLabe.width, bulletLabel.height);
    } completion:^(BOOL finished) {
        //
}];

这样实现了一个弹幕移动的过程。借助于系统 UIView 动画,实现一个滚动过程中,开发者只需要关注动画开始时间、持续时间、动画开始执行的状态和动画结束执行的状态,这里采用的 UILabel frame。具体的滚动动画交于系统来负责,系统会根据设备的性能来调整来做对应的调整。

弹幕如何控制移动速度

首先根据文本的内容的属性(字体、大小)计算出文本显示后的 Width,line 为 1,根据 textWidth 及 screenWidth 来却确定最终的 rate。

计算文本 width 可通过 boundingRect(with:options:attributes:context:)

弹幕如何检测是否碰撞

检测难点在于,怎么在弹幕滚动过程中检测是否碰撞。弹幕的滚动过程中位于同一个 Y 的弹幕,新创建的弹幕显示 frame 需要与正在滚动中的弹幕 frame 进行对照。如果新创建的弹幕滚动的过程中,不与正在滚动中的弹幕有视图重叠。则新创建的弹幕以当前 frame 作为初始位置开始滚动。

那么两者如何进行对照呢?

默认弹幕由右向左滚动,过程中,正在滚动中的弹幕位于左侧,新建的弹幕位于右侧。

如果正在滚动中的弹幕(记为 danmakuL)的开始时刻及结束时刻,弹幕视图都没有与新建弹幕(记为 danmakuR)视图重合。那么新建弹幕在滚动过程中也不会与正在滚动中的弹幕有视图重合的情况出现。

remainTime 代表着字幕存活时间,由于字幕的存活时间是由字幕的长度解决的。字幕越长,滚动过程中速度即越快。所以过程中已 danmakuL 与 danmakuR 间最小滚动时间(miniRemainTime)为标准,进行对比。如果在 miniRemainTime 内,danmakuL 与 danmakuR 之间没有出现重合,那么整个过程中,弹幕也不会出现重合的情况。这样就解决了弹幕碰撞的检测。

演示代码如下:

- (BOOL)checkIsWillHitWithWidth:(float)width danmakuL:(UILabel *)danmakuL danmakuR:(UILabel *)danmakuR {
    if (danmakuL.remainTime<=0) {
        return NO;
    }
    
    if (danmakuL.x + danmakuL.size.width > danmakuR.x) {
        return YES;
    }
    
    float minRemainTime = MIN(danmakuL.remainTime, danmakuR.remainTime);
    float xLeft = [danmakuL xWithScreenWidth:width remainTime:(danmakuL.remainTime - minRemainTime)];
    float xRight = [danmakuR xWithScreenWidth:width remainTime:(danmakuR.remainTime - minRemainTime)];
    if (xLeft + danmakuL.size.width > xRight) {
        return YES;
    }
    return NO;
}

弹幕如何暂停滚动及恢复滚动

在当前的实现过程中,弹幕滚动采用了 UIView 系统动画。动画暂停的基本原理是通过 View 的 presentationLayer 获取 danmake 当前 frame 并作为弹幕暂停后的 frame,再移除 danmake 的 layer 动画。同时记录已滚动时间及当前 danmake frame。


// 暂停弹幕滚动
- (void)pauseDanmaku {
    
    CALayer *layer = danmaku.layer;
    CGRect rect = danmaku.frame;
    danmaku.label.frame = rect;
    [danmaku.label.layer removeAllAnimations];
}

// 开始滚动的方法与新建弹幕滚动方法相同,通过 UIView animation
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容