本篇是swift版本的弹幕,原理同我的上篇OC版本的弹幕检测
ios 弹幕 网上找了好多弹幕demo,发现很多都会重叠,体验很不好。所以在参考部分网上弹幕源码的基础上,遂有了本篇弹幕碰撞检测。
本弹幕优点如下(这个是最初版本,以后慢慢优化):
一. 可以很好的避免弹幕间的碰撞
其实针对碰撞,主要解决水平运动时,前后两个弹幕不碰撞就可以了。
所以我们主要针对同一个轨道上,前后弹幕是否碰撞展开研究,就可以了。
倘若刚刚生产的弹幕,在同一轨道上会发生碰撞,我们就选择下一个轨道再进行判断,直到找到不发生碰撞的轨道。(此处可能会有极少弹幕丢失,以后再优化)
当前一个弹幕的运行时间变成0时(此处我们默认每一个弹幕的运行时间是5s),也就说明前一个弹幕已经移动到屏幕外面了,后一个弹幕可以发射了。 2.当前一个弹幕的运行时间大于0时,我们判断前一个弹幕是否完全进入屏幕,倘如完全进入屏幕的话,接着判断后一个弹幕是否能追得上前一个弹幕(是否发生碰撞)。
3.当前一个弹幕的运行时间大于0时,我们判断前一个弹幕是否完全进入屏幕,倘如未完全进入屏幕的话,判断下一个轨道(从1开始)。 具体判断代码如下: 我们假定弹幕的运行时间是5s,弹幕向前走一步,时间减少0.2s。
主要逻辑代码如下:// MARK: 弹幕碰撞检测 YES:会碰撞 NO:不会碰撞 func judgeHitWithPreDanmaku(_ preDanmaku : DanmuView,_ danmaku : DanmuView) -> Bool { if preDanmaku.remainingTime != nil { //1.前一个弹幕是否还在移动?【显示时间每次递减0.2s】 if (preDanmaku.remainingTime! <= Float(0.0)) { return false; //说明前一个弹幕已经移出了屏幕,不会碰撞 } //屏幕的宽度 let width = self.bounds.width; //5s显示时间下的弹幕移动速度 let preDanmakuSpeed = (width + (preDanmaku.size?.width)!) / CGFloat(Duration); //2.已经移入屏幕的距离与弹幕要移动的总距离比较 if (preDanmakuSpeed * (CGFloat(Duration) - CGFloat(preDanmaku.remainingTime!)) < (preDanmaku.size?.width)!) { return true; //说明弹幕未完全进入屏幕,只有一部分进入了屏幕,会发生碰撞 } //3.当前弹幕能否追得上前一个弹幕? let currentDanmakuSpeed = (width + (danmaku.size?.width)!) / CGFloat(Duration); if (currentDanmakuSpeed * CGFloat(preDanmaku.remainingTime!) > width) { return true; //可以追得上,会发生碰撞 } return false; } return false; }
swiftdemo截图:
原文链接:https://www.f2er.com/swift/320956.html