在 iOS 中实现谷歌灭霸彩蛋

2024-02-20 19:59
文章标签 实现 谷歌 ios 彩蛋 灭霸

本文主要是介绍在 iOS 中实现谷歌灭霸彩蛋,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

示例代码下载

最近上映的复仇者联盟4据说没有片尾彩蛋,不过谷歌帮我们做了。只要在谷歌搜索灭霸,在结果的右侧点击无限手套,你将化身为灭霸,其中一半的搜索结果会化为灰烬消失…那么这么酷的动画在iOS中可以实现吗?答案是肯定的。整个动画主要包含以下几部分:响指动画、沙化消失以及背景音效和复原动画,让我们分别来看看如何实现。

在 iOS 中实现谷歌灭霸彩蛋

在 iOS 中实现谷歌灭霸彩蛋

图1 上为沙化动画,下为复原动画

 

响指动画

 


Google的方法是利用了48帧合成的一张Sprite图进行动画的:

image.png

图2 响指Sprite图片

原始图片中48幅全部排成一行,这里为了显示效果截成2行

iOS 中通过这张图片来实现动画并不难。CALayer有一个属性contentsRect,通过它可以控制内容显示的区域,而且是Animateable的。它的类型是CGRect,默认值为(x:0.0, y:0.0, width:1.0, height:1.0),它的单位不是常见的Point,而是单位坐标空间,所以默认值显示100%的内容区域。新建Sprite播放视图层AnimatableSpriteLayer:

class AnimatableSpriteLayer: CALayer {private var animationValues = [CGFloat]()convenience init(spriteSheetImage: UIImage, spriteFrameSize: CGSize ) {self.init()//1masksToBounds = truecontentsGravity = CALayerContentsGravity.leftcontents = spriteSheetImage.cgImagebounds.size = spriteFrameSize//2let frameCount = Int(spriteSheetImage.size.width / spriteFrameSize.width)for frameIndex in 0..<frameCount {animationValues.append(CGFloat(frameIndex) / CGFloat(frameCount))}}func play() {let spriteKeyframeAnimation = CAKeyframeAnimation(keyPath: "contentsRect.origin.x")spriteKeyframeAnimation.values = animationValuesspriteKeyframeAnimation.duration = 2.0spriteKeyframeAnimation.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.linear)//3spriteKeyframeAnimation.calculationMode = CAAnimationCalculationMode.discreteadd(spriteKeyframeAnimation, forKey: "spriteKeyframeAnimation")}
}

//1: masksToBounds = true和contentsGravity = CALayerContentsGravity.left是为了当前只显示Sprite图的第一幅画面

//2: 根据Sprite图大小和每幅画面的大小计算出画面数量,预先计算出每幅画面的contentsRect.origin.x偏移量

//3: 这里是关键,指定关键帧动画的calculationMode为discrete确保关键帧动画依次使用values中指定的关键帧值进行变化,而不是默认情况下采用线性插值进行过渡,来个对比图可能比较容易理解:

在 iOS 中实现谷歌灭霸彩蛋在 iOS 中实现谷歌灭霸彩蛋

图3 左边为离散模式,右边为默认的线性模式

 

沙化消失

 


这个效果是整个动画较难的部分,Google的实现很巧妙,它将需要沙化消失内容的html通过html2canvas渲染成canvas,然后将其转换为图片后的每一个像素点随机地分配到32块canvas中,最后对每块画布进行随机地移动和旋转即达到了沙化消失的效果。

像素处理

新建自定义视图 DustEffectView,这个视图的作用是用来接收图片并将其进行沙化消失。首先创建函数createDustImages,它将一张图片的像素随机分配到32张等待动画的图片上:

class DustEffectView: UIView {private func createDustImages(image: UIImage) -> [UIImage] {var result = [UIImage]()guard let inputCGImage = image.cgImage else {return result}//1let colorSpace = CGColorSpaceCreateDeviceRGB()let width = inputCGImage.widthlet height = inputCGImage.heightlet bytesPerPixel = 4let bitsPerComponent = 8let bytesPerRow = bytesPerPixel * widthlet bitmapInfo = CGImageAlphaInfo.premultipliedLast.rawValue | CGBitmapInfo.byteOrder32Little.rawValueguard let context = CGContext(data: nil, width: width, height: height, bitsPerComponent: bitsPerComponent, bytesPerRow: bytesPerRow, space: colorSpace, bitmapInfo: bitmapInfo) else {return result}context.draw(inputCGImage, in: CGRect(x: 0, y: 0, width: width, height: height))guard let buffer = context.data else {return result}let pixelBuffer = buffer.bindMemory(to: UInt32.self, capacity: width * height)//2let imagesCount = 32var framePixels = Array(repeating: Array(repeating: UInt32(0), count: width * height), count: imagesCount)for column in 0..<width {for row in 0..<height {let offset = row * width + column//3for _ in 0...1 { let factor = Double.random(in: 0..<1) + 2 * (Double(column)/Double(width))let index = Int(floor(Double(imagesCount) * ( factor / 3)))framePixels[index][offset] = pixelBuffer[offset]}}}//4for frame in framePixels {let data = UnsafeMutablePointer(mutating: frame)guard let context = CGContext(data: data, width: width, height: height, bitsPerComponent: bitsPerComponent, bytesPerRow: bytesPerRow, space: colorSpace, bitmapInfo: bitmapInfo) else {continue}result.append(UIImage(cgImage: context.makeImage()!, scale: image.scale, orientation: image.imageOrientation))}return result}
}

//1: 根据指定格式创建位图上下文,然后将输入的图片绘制上去之后获取其像素数据

//2: 创建像素二维数组,遍历输入图片每个像素,将其随机分配到数组32个元素之一的相同位置。随机方法有点特别,原始图片左边的像素只会分配到前几张图片,而原始图片右边的像素只会分配到后几张。

在 iOS 中实现谷歌灭霸彩蛋

图4 上部分为原始图片,下部分为像素分配后的32张图片依次显示效果

//3: 这里循环2次将像素分配两次,可能 Google 觉得只分配一遍会造成像素比较稀疏。个人认为在移动端,只要一遍就好了。

//4: 创建32张图片并返回

添加动画

Google的实现是给canvas中css的transform属性设置为rotate(deg) translate(px, px) rotate(deg),值都是随机生成的。如果你对CSS的动画不熟悉,那你会觉得在iOS中只要添加三个CABasicAnimation然后将它们添加到AnimationGroup就好了嘛,实际上并没有那么简单… 因为CSS的transform中后一个变换函数是基于前一个变换后的新transform坐标系。假如某张图片的动画样式是这样的:rotate(90deg) translate(0px, 100px) rotate(-90deg) 直觉告诉我应该是旋转着向下移动100px,然而在CSS中的元素是这么运动的:

在 iOS 中实现谷歌灭霸彩蛋

图5 CSS中transform多值动画 

第一个rotate和translate决定了最终的位置和运动轨迹,至于第二个rotate作用,只是叠加第一个rotate的值作为最终的旋转弧度,这里刚好为0也就是不旋转。那么在iOS中该如何实现相似的运动轨迹呢?可以利用UIBezierPath, CAKeyframeAnimation的属性path可以指定这个UIBezierPath为动画的运动轨迹。确定起点和实际终点作为贝塞尔曲线的起始点和终止点,那么如何确定控制点?好像可以将“预想”的终点(下图中的(0,-1))作为控制点。

image.png

图6 将“预想”的终点作为控制点的贝塞尔曲线,看起来和CSS中的运动轨迹差不多 

扩展问题

通过文章中描述的方式生成的贝塞尔曲线是否与CSS中的动画轨迹完全一致呢?

现在可以给视图添加动画了:

    let layer = CALayer()layer.frame = boundslayer.contents = image.cgImageself.layer.addSublayer(layer)let centerX = Double(layer.position.x)let centerY = Double(layer.position.y)let radian1 = Double.pi / 12 * Double.random(in: -0.5..<0.5)let radian2 = Double.pi / 12 * Double.random(in: -0.5..<0.5)let random = Double.pi * 2 * Double.random(in: -0.5..<0.5)let transX = 60 * cos(random)let transY = 30 * sin(random)//1: // x' = x*cos(rad) - y*sin(rad)// y' = y*cos(rad) + x*sin(rad)let realTransX = transX * cos(radian1) - transY * sin(radian1)let realTransY = transY * cos(radian1) + transX * sin(radian1)let realEndPoint = CGPoint(x: centerX + realTransX, y: centerY + realTransY)let controlPoint = CGPoint(x: centerX + transX, y: centerY + transY)//2:let movePath = UIBezierPath()movePath.move(to: layer.position)movePath.addQuadCurve(to: realEndPoint, controlPoint: controlPoint)let moveAnimation = CAKeyframeAnimation(keyPath: "position")moveAnimation.path = movePath.cgPathmoveAnimation.calculationMode = .paced//3:                 let rotateAnimation = CABasicAnimation(keyPath: "transform.rotation")rotateAnimation.toValue = radian1 + radian2let fadeOutAnimation = CABasicAnimation(keyPath: "opacity")fadeOutAnimation.toValue = 0.0let animationGroup = CAAnimationGroup()animationGroup.animations = [moveAnimation, rotateAnimation, fadeOutAnimation]animationGroup.duration = 1//4:animationGroup.beginTime = CACurrentMediaTime() + 1.35 * Double(i) / Double(imagesCount)animationGroup.isRemovedOnCompletion = falseanimationGroup.fillMode = .forwardslayer.add(animationGroup, forKey: nil)

//1: 实际的偏移量旋转了radian1弧度,这个可以通过公式x' = x*cos(rad) – y*sin(rad), y' = y*cos(rad) + x*sin(rad)算出

//2: 创建UIBezierPath并关联到CAKeyframeAnimation中

//3: 两个弧度叠加作为最终的旋转弧度

//4: 设置CAAnimationGroup的开始时间,让每层Layer的动画延迟开始

结尾

 


到这里,谷歌灭霸彩蛋中较复杂的技术点均已实现。如果您感兴趣,完整的代码(包含音效和复原动画)可以通过文章开头的链接进行查看,可以尝试将沙化图片的数量从32提高至更多,效果会越好,内存也会消耗更多 :-D。

 

作者:potato04  作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要这是一个我的iOS交流群:869685378,不管你是小白还是大牛欢迎入驻 ,分享BAT,阿里面试题、面试经验,讨论技术, 大家一起交流学习成长!

这篇关于在 iOS 中实现谷歌灭霸彩蛋的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/729375

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现