CSS3近阶段篇之酷炫的3D旋转透视


Posted in HTML / CSS onApril 28, 2016

之前学习 react+webpack ,偶然路过 webpack 官网 ,看到顶部的 LOGO ,就很感兴趣。

最近觉得自己 CSS3 过于薄弱,想着深入学习一番,遂以这个 LOGO 为切入口,好好研究学习了一下相关的 CSS3 属性。webpack 的 LOGO 动画效果乍看不是很难,深入了解之后,觉得内部其实大有学问,自己折腾了一番,做了一系列相关的 CSS3 动画效果。

先上 demo ,没有将精力放在兼容上,请用 chrome 打开。

本文完整的代码,以及更多的 CSS3 效果,在我github上可以看到,也希望大家可以点个 star。

嗯,可能有些人打不开 demo 或者页面乱了,贴几张效果图:(图片有点大,耐心等待一会)

立方体 3D 旋转

CSS3近阶段篇之酷炫的3D旋转透视

3D 透视照片墙

CSS3近阶段篇之酷炫的3D旋转透视

跳跃的音符

CSS3近阶段篇之酷炫的3D旋转透视

可能上面的效果对精通 CSS3 的而言小菜一碟,写本文的目的也是我自己学习积累的一个过程,感兴趣的就可以一起往下看啦。

其实 CSS3 效果真的很强大,上面的效果都是纯 CSS 实现,个人感觉越是深入 CSS 的学习,越是觉得自己不懂 CSS ,不过话说回来,这些效果的实用场景不大,但是作为一个有追求的前端,我觉得还是有必要去好好了解一下这些属性。

所以本文接下来要讲的大概有:

transform-style:preserve-3d三维效果perspective andperspective-origin 3D视距,透视/景深效果CSS3filter CSS3滤镜transparent、radial-gradient 透明与渐变

transform-style

要利用 CSS3 实现 3D 的效果,最主要的就是借助transform-style 属性。transform-style 只有两个值可以选择:

CSS Code复制内容到剪贴板
  1. // 语法:   
  2. transform-style: flat|preserve-3d;   
  3. transform-style: flat; // 默认,子元素将不保留其 3D 位置   
  4. transform-style: preserve-3d; // 子元素将保留其 3D 位置。  

当我们指定一个容器的transform-style 的属性值为preserve-3d 时,容器的后代元素便会具有 3D 效果,这样说有点抽象,也就是当前父容器设置了preserve-3d 值后,它的子元素就可以相对于父元素所在的平面,进行 3D 变形操作。

当父元素设置了 transform-style:preserve-3d 后,就可以对子元素进行 3D 变形操作了,3D 变形和2D 变形一样可以,使用 transform 属性来设置,或者可以通过制定的函数或者通过三维矩阵来对元素变型操作:

1、使用translateX(length) 、translateY(length) 、 translateZ(length) 来进行 3D 位移操作,与 2D 操作一样,对元素进行位移操作,也可以合并为translate3d(x,y,z) 这种写法;

2、使用scaleX() 、scaleY() 、scaleY() 来进行3D 缩放操作,也可以合并为 scale3d(number,number,number)这种写法;

3、使用rotateX(angle) 、rotateY(angle) 、rotateZ(angle) 来进行 3D 旋转操作,也可以合并为rotate3d(Xangle,Yangle,Zangle) 这种写法。

对于 API 的学习,我建议去源头看看,不要满足于消费别人的总结,transform-style API。

这里要特别提出的,3D 坐标轴,所谓的绕 X、Y、Z 轴的三个轴,这个不难,感觉空间想象困难的,照着 API 试试,绕每个轴都转一下就明白了:

CSS3近阶段篇之酷炫的3D旋转透视

了解过后,那么依靠上面所说的,其实我们就已经可以做一个立方体出来了。所谓实践出真知,下面就看看该如何一步步得到一个立方体。

1、准备六个正方形

这个好理解,正方体六个面,首先用 div 做出 6 个面,包裹在同一个父级容器下面,父级容器设置transform-style:preserve-3d ,这样接下来就可以对 6 个面进行 3D 变换操作,为了方便演示,我用 6 个颜色不一样的面:

CSS3近阶段篇之酷炫的3D旋转透视

上面的图是示意有 6 个面,当然我们要把 6 个正方形 div 设置为绝对定位,重叠叠在一起,那么应该是这样的,只能看到一个面:

CSS3近阶段篇之酷炫的3D旋转透视

2、父容器做简单的变换

为了最后的看上去的效果好看,我们需要先对父容器进行变换,运用上面说的rotate 属性,将容器的角度改变一下:

CSS Code复制内容到剪贴板
  1. .cube-container{   
  2.     -webkit-transform: rotateX(-33.5deg) rotateY(45deg);   
  3.     transform: rotateX(-33.5deg) rotateY(45deg);   
  4. }  

那么,变换之后,得到这样一个图形:

CSS3近阶段篇之酷炫的3D旋转透视

嗯,这个时候,6 个 div 面仍然是重叠在一起的。

3、对每个面做 3D 变换

接下来就是对每个面进行 3D 变换了,运用 rotate 可以对div 平面进行旋转,运用 translate 可以对 div 平面进行平移,而且要记住现在我们是在三维空间内变换,转啊转啊,我们可能会得到这样的形状:

CSS3近阶段篇之酷炫的3D旋转透视

算好旋转角度和偏移距离,最后上面的 6 个面就可以完美拼成一个立方体咯!为了效果更好,我给每个面增加一些透明度,最后得到一个完整的立方体:

CSS3近阶段篇之酷炫的3D旋转透视

为了更有立体感,我们可以调整父容器的旋转角度,旋转看上去更立体的角度:

CSS3近阶段篇之酷炫的3D旋转透视

至此,一个 3D 立方体就完成了。写这篇文章的时候,本来到这里,这一块应该就结束了,但是写到这里的时候,突然突发奇想,既然正方体可以(正六面体),那么正四面体,正八面体甚至球体应该也能做出来吧?

嗯,没有按住躁动的心,立马动手尝试了一番,最后做出了下面的两个:

CSS3近阶段篇之酷炫的3D旋转透视

CSS3近阶段篇之酷炫的3D旋转透视

就不再详细讨论如何一步一步得到这两个了,有兴趣的可以去我的 github 上看看源码,或者直接和我讨论交流,简单的谈谈思路:

正四面体

和正方体一样,我们首先要准备 4 个三角形(下面会详细讲如何利用 CSS3 制作一个三角形 div),注意 4 个三角形应该是重叠在一起的,然后将其中三个分别沿着三条边的中心点旋转 70.5 度(正四面体临面夹角),就可以得到一个正四面体。

注意沿着三条边的中心点旋转 70.5 度这句话,意思是每个图形要定位一次旋转中心,也就是transform-origin 属性,它允许我们设置旋转元素的基点位置。

球体

上面的 GIF 图因为添加了 animation 动画效果,看上去很像一个球体在运动,其实只用了 4 个 div,每个 div 利用 border-radius:100% 设置为圆形,然后以中心点为基准,每个圆形 div 绕 Y 轴旋转不同的角度,再让整个圆形容器绕 Y 轴动起来,就可以得到这样一个效果了。

perspective andperspective-origin 3D视距,透视/景深效果

继续说 3D ,接下来要说的属性是 persepective 和 perspective-origin 。

persepective

CSS Code复制内容到剪贴板
  1. // 语法   
  2. perspective: number|none;  

perspective 为一个元素设置三维透视的距离,仅作用于元素的后代,而不是其元素本身。

简单来说,当元素没有设置 perspective时,也就是当 perspective:none/0 时所有后代元素被压缩在同一个二维平面上,不存在景深的效果。

而如果设置 perspective后,将会看到三维的效果。

perspective-origin

perspective-origin 表示3D 元素透视视角的基点位置,默认的透视视角中心在容器是 perspective 所在的元素,而不是他的后代元素的中点,也就是 perspective-origin: 50% 50%。

CSS Code复制内容到剪贴板
  1. // 语法   
  2. perspective-origin: x-axis y-axis;   
  3. // x-axis : 定义该视图在 x 轴上的位置。默认值:50%   
  4. // y-axis : 定义该视图在 y 轴上的位置。默认值:50%  

值得注意的是,CSS3 3D 变换中的透视的透视点是在浏览器的前方。

说总是很难理解,运用上面我们做出来的正方体试验一下,我设置了正方体的边长为 50 px ,这里设置正方体容器 cuber-inner 的persepective 的为 100 px,而 perspective-origin 保持为默认值:

CSS Code复制内容到剪贴板
  1. -webkit-perspective-origin: 50% 50%;   
  2. perspective-origin: 50% 50%;   
  3. -webkit-perspective: 100px;   
  4. perspective: 100px;  

上面这样设置,也就是相当于我站在 100px 的长度外去看这个立方体,效果如下:

CSS3近阶段篇之酷炫的3D旋转透视

通过调整persepective 和perspective-origin 的值,可以看到不一样的图形,这个很好理解,我们观测一个物体的角度和距离物体的距离不断发生改变,我们看的物体也是不一样的,嗯想象一下小学课文,杨桃和星星,就能容易明白了。

需要提出的是,我上面的几个正多面体图形和球形图形是没有加上persepective 值的,感兴趣的可以加上试一下看看效果。

3D 透视照片墙

回到文章一开始我贴的那个 3D 照片墙,运用transform-style: preserve-3d 和persepective ,可以做出效果很好的这种 3D 照片墙旋转效果。

代码就不贴了,本文已经很长了,只是简单的谈谈原理,感兴趣的去扒源码看看。

1、设立一个舞台,也就是包裹旋转的图片们的容器,给他设置一个persepective 距离,以及transform-style: preserve-3d 让后代可以进行 3D 变换;

2、准备 N 张图片置于容器内部,N 的大小看个人喜好了,图片的 3D 旋转木马效果是类似钢管舞旋转的运动,因此是绕 Y 轴的,我们关心的是rotateY的大小,根据我们添加的图片数量,用 360° 的圆周角将每个图片等分,也就是让每张图片绕 Y 轴旋转固定角度依次散开:(下面的图为示意效果,我调整了一下角度和透明度)

CSS3近阶段篇之酷炫的3D旋转透视

3、这个时候,N 张图肯定是重合叠在了一起,所以这里关键一步是运用translateZ(length) 让图片沿 Z 轴平移,也就是运用translateZ 可以让图片离我们更近或者更远,因为上一步设置了图片不同的rotateY() 角度,所以 N 张图片设定一个translateZ 后,图片就很自然以中点为圆心分散开了,也就是这样:

CSS3近阶段篇之酷炫的3D旋转透视

4、最后利用 animation ,我们让舞台,也就是包裹着图片的容器绕 Y 轴旋转起来(rotateY),那么一个类似旋转木马的 3D 照片墙效果就完成了!

这里要注意的一点是设置的persepective 值和单个图片translateZ(length) 的值,persepective 一定要比translateZ(length) 的值大,否则就是相当于舞台跑你身后去了,肯定是看不到效果了。

HTML / CSS 相关文章推荐
CSS3 简单又实用的5个属性
Mar 04 HTML / CSS
CSS3感应鼠标的背景闪烁和图片缩放动画效果
May 14 HTML / CSS
HTML5+CSS3实例 :canvas 模拟实现电子彩票刮刮乐代码
Dec 30 HTML / CSS
CSS3 transition 实现通知消息轮播条
Oct 14 HTML / CSS
html5 浏览器支持 如何让所有的浏览器都支持HTML5标签样式
Dec 07 HTML / CSS
HTML5实现获取地理位置信息并定位功能
Apr 25 HTML / CSS
在HTML5中使用MathML数学公式的简单讲解
Feb 19 HTML / CSS
实例讲解使用SVG制作loading加载动画的方法
Apr 05 HTML / CSS
HTML5中的强制下载属性download使用实例解析
May 12 HTML / CSS
HTML5中图片之间的缝隙完美解决方法
Jul 07 HTML / CSS
html5开发三八女王节表白神器
Mar 07 HTML / CSS
使用canvas来完成线性渐变和径向渐变的功能的方法示例
Jul 25 HTML / CSS
CSS3的first-child选择器实战攻略
Apr 28 #HTML / CSS
CSS3径向渐变之大鱼吃小鱼之孤单的大鱼
Apr 26 #HTML / CSS
CSS3实现超酷的黑猫警长首页
Apr 26 #HTML / CSS
CSS3实现千变万化的文字阴影text-shadow效果设计
Apr 26 #HTML / CSS
什么是CSS3 HSLA色彩模式?HSLA模拟渐变色条
Apr 26 #HTML / CSS
CSS3 RGBA色彩模式使用实例讲解
Apr 26 #HTML / CSS
CSS3不透明度实例讲解
Apr 26 #HTML / CSS
You might like
PHP大批量数据操作时临时调整内存与执行时间的方法
2011/04/20 PHP
php防止CC攻击代码 php防止网页频繁刷新
2015/12/21 PHP
老生常谈文本文件和二进制文件的区别
2017/02/27 PHP
PHP结合Redis+MySQL实现冷热数据交换应用案例详解
2019/07/09 PHP
Laravel项目中timeAgo字段语言转换的改善方法示例
2019/09/16 PHP
JavaScript网页制作特殊效果用随机数
2007/05/22 Javascript
JavaScript 浏览器验证代码(来自discuz)
2010/07/17 Javascript
javascript获取隐藏dom的宽高 具体实现
2013/07/14 Javascript
JS+CSS实现下拉列表框美化效果(3款)
2015/08/15 Javascript
JavaScript学习笔记(三):JavaScript也有入口Main函数
2015/09/12 Javascript
jQuery实现分隔条左右拖动功能
2015/11/21 Javascript
深入理解Ajax的get和post请求
2016/06/02 Javascript
JavaScript定义数组的三种方法(new Array(),new Array('x','y')
2016/10/04 Javascript
js实现手机发送验证码功能
2017/03/13 Javascript
详解Angular 自定义结构指令
2017/06/21 Javascript
vue-quill-editor实现图片上传功能
2017/08/08 Javascript
详解weex默认webpack.config.js改造
2018/01/08 Javascript
AngularJS日期格式化常见操作实例分析
2018/05/17 Javascript
微信小程序实现页面下拉刷新和上拉加载功能详解
2018/12/03 Javascript
layer.open提交子页面的form和layedit文本编辑内容的方法
2019/09/27 Javascript
vue实现二级导航栏效果
2019/10/19 Javascript
vue 页面回退mounted函数不执行的解决方案
2020/07/26 Javascript
解决vue与node模版引擎的渲染标记{{}}(双花括号)冲突问题
2020/09/11 Javascript
[01:50]2014DOTA2西雅图邀请赛 专访欢乐周宝龙
2014/07/08 DOTA
[01:16:28]DOTA2-DPC中国联赛 正赛 iG vs Magma BO3 第二场 2月23日
2021/03/11 DOTA
python3使用requests模块爬取页面内容的实战演练
2017/09/25 Python
python 定时修改数据库的示例代码
2018/04/08 Python
毕业生自我鉴定
2013/11/05 职场文书
机械设计毕业生自荐信
2014/02/02 职场文书
道德之星事迹材料
2014/05/03 职场文书
工伤事故赔偿协议书(标准)
2014/09/29 职场文书
基层党员群众路线教育实践活动个人对照检查材料思想汇报
2014/10/05 职场文书
学校党的群众路线教育实践活动总结材料
2014/10/30 职场文书
写给媳妇的检讨书
2015/05/06 职场文书
终止解除劳动合同证明书
2015/06/17 职场文书
《巫师》是美食游戏?CDPR10月将推出《巫师》官方食谱
2022/04/03 其他游戏