CSS3 3D 技术手把手教你玩转


Posted in Javascript onSeptember 02, 2016

css3的3d起步

要玩转css3的3d,就必须了解几个词汇,便是透视(perspective)、旋转(rotate)和移动(translate)。透视即是以现实的视角来看屏幕上的2D事物,从而展现3D的效果。旋转则不再是2D平面上的旋转,而是三维坐标系的旋转,就包括X轴,Y轴,Z轴旋转。平移同理。

当然用理论来说明,估计你还不明白。下面是3个gif:

沿着X轴旋转

CSS3 3D 技术手把手教你玩转

沿着Y轴旋转

CSS3 3D 技术手把手教你玩转

沿着Z轴旋转

CSS3 3D 技术手把手教你玩转

旋转应该没问题了,那理解平移起来就比较容易了,就是在在X轴、Y轴、z轴移动。

你可能会说透视比较不好理解,下面我介绍一下透视的几个属性。

perspective

perspective英文名便是透视,没有这东西就没办法形成3D效果,但是这个东西是怎么让我们浏览器形成3D 效果的呢,可以看下面这张图,其实学过绘画的应该知道透视关系,而这里就是这个道理。

CSS3 3D 技术手把手教你玩转

但是在css里它是有数值的,例如perspective: 1000px这个代表什么呢?我们可以这样理解,如果我们直接眼睛靠着物体看,物体就超大占满我们的视线,我们离它距离越来越大,它在变小,立体感也就出来了是不是,其实这个数值构造了一个我们眼睛离屏幕的距离,也就构造了一个虚拟3D假象。

perspective-origin

由上面我们了解了perspective,而加上了这个origin是什么,我们前面说的这个是眼睛离物体的距离,而这个就是眼睛的视线,我们的视点的不同位置就决定了我们看到的不同景象,默认是中心,为perspectice-origin: 50% 50%,第一个数值是 3D 元素所基于的 X 轴,第二个定义在 y 轴上的位置

当为元素定义 perspective-origin 属性时,其子元素会获得透视效果,而不是元素本身。必须与 perspective 属性一同使用,而且只影响 3D 转换元素。(W3school)

CSS3 3D 技术手把手教你玩转

transform-style

perspective又来了,没错,它是css中3D的关键,transform-style默认是flat,如果你要在元素上视线3D效果的话,就必须用上transform-style: preserve-3d,否则就只是平面的变换,而不是3D的变换

手把手带你玩转css3-3d

以上我们稍微了解概念,下面就开始实战吧!

我们看一个效果,是不是很酷炫:直接访问https://bupt-hjm.github.io/css3-3d/,觉得可以的话记得给star咯hh

第一步:html结构

很简单的一个容器包裹着一个装了6个piece的piece-box

<div class="container">
 <div class="piece-box">
  <div class="piece piece-1"></div>
  <div class="piece piece-2"></div>
  <div class="piece piece-3"></div>
  <div class="piece piece-4"></div>
  <div class="piece piece-5"></div>
  <div class="piece piece-6"></div>
 </div>
</div>

第二步: 加上必要的3D属性,进入3D世界

通过上面的讲解,应该对perspective比较熟悉了吧,

/*容器*/
.container {
 -webkit-perspective: 1000px;
 -moz-perspective: 1000px;
 -ms-perspective: 1000px;
 perspective: 1000px;
}
/*piece盒子*/
 .piece-box {
  perspective-origin: 50% 50%;
  -webkit-transform-style: preserve-3d;
  -moz-transform-style: preserve-3d;
  -ms-transform-style: preserve-3d;
  transform-style: preserve-3d;
}

第三步:加入必要的样式

/*容器*/
.container {
 -webkit-perspective: 1000px;
 -moz-perspective: 1000px;
 -ms-perspective: 1000px;
 perspective: 1000px;
}
/*piece盒子*/
.piece-box {
 position: relative;
 width: 200px;
 height: 200px;
 margin: 300px auto;
 perspective-origin: 50% 50%;
 -webkit-transform-style: preserve-3d;
 -moz-transform-style: preserve-3d;
 -ms-transform-style: preserve-3d;
 transform-style: preserve-3d;
}
/*piece通用样式*/
.piece {
 position: absolute;
 width: 200px;
 height: 200px;
 background: red;
 opacity: 0.5;

}
.piece-1 {
 background: #FF6666;

}
.piece-2 {
 background: #FFFF00;
}
.piece-3 {
 background: #006699;
}
.piece-4 {
 background: #009999;
}
.piece-5 {
 background: #FF0033;
}
.piece-6 {
 background: #FF6600;
}

当然,在你完成这步之后你还是只看到一个正方形,也就是我们的piece-6,因为我们的3Dtransform还没开始

CSS3 3D 技术手把手教你玩转

第四步:3D变换来袭

首先是实现走马灯,我们先不要让它走,先实现灯部分。

如何实现呢?因为要构成一个圆,圆是360度,而我们有6个piece,那么,很容易想到,我们需要把每一个piece以递增60度的方式rotateY,就变成如下

CSS3 3D 技术手把手教你玩转

如何把他们从中心拉开呢?

这里我们还要注意一点,在我们使元素绕Y轴旋转以后,其实X轴和Z轴也会跟着旋转,所所以正方体的每个面的垂直线还是Z轴,我们就只需要改变下translateZ的值,而当translateZ为正的时候,就朝着我们的方向走来,这样就可以拉开了

但是拉开的距离如何衡量?

CSS3 3D 技术手把手教你玩转

是不是一目了然了~

下面我们再修改下css

.piece-1 {
 background: #FF6666;
 -webkit-transform: rotateY(0deg) translateZ(173.2px);
 -ms-transform: rotateY(0deg) translateZ(173.2px);
 -o-transform: rotateY(0deg) translateZ(173.2px);
 transform: rotateY(0deg) translateZ(173.2px);

}
.piece-2 {
 background: #FFFF00;
 -webkit-transform: rotateY(60deg) translateZ(173.2px);
 -ms-transform: rotateY(60deg) translateZ(173.2px);
 -o-transform: rotateY(60deg) translateZ(173.2px);
 transform: rotateY(60deg) translateZ(173.2px);
}
.piece-3 {
 background: #006699;
 -webkit-transform: rotateY(120deg) translateZ(173.2px);
 -ms-transform: rotateY(120deg) translateZ(173.2px);
 -o-transform: rotateY(120deg) translateZ(173.2px);
 transform: rotateY(120deg) translateZ(173.2px);
}
.piece-4 {
 background: #009999;
 -webkit-transform: rotateY(180deg) translateZ(173.2px);
 -ms-transform: rotateY(180deg) translateZ(173.2px);
 -o-transform: rotateY(180deg) translateZ(173.2px);
 transform: rotateY(180deg) translateZ(173.2px);
}
.piece-5 {
 background: #FF0033;
 -webkit-transform: rotateY(240deg) translateZ(173.2px);
 -ms-transform: rotateY(240deg) translateZ(173.2px);
 -o-transform: rotateY(240deg) translateZ(173.2px);
 transform: rotateY(240deg) translateZ(173.2px);
}
.piece-6 {
 background: #FF6600;
 -webkit-transform: rotateY(300deg) translateZ(173.2px);
 -ms-transform: rotateY(300deg) translateZ(173.2px);
 -o-transform: rotateY(300deg) translateZ(173.2px);
 transform: rotateY(300deg) translateZ(173.2px);
}

是不是已经实现了走马灯了?

CSS3 3D 技术手把手教你玩转

第五步:animation让3D动起来

要实现走马灯动,其实很简单,我们只要在piece-box上加上旋转动画就行了,5s完成动画,从0度旋转到360度

/*piece盒子*/
.piece-box {
 position: relative;
 width: 200px;
 height: 200px;
 margin: 300px auto;
 perspective-origin: 50% 50%;
 -webkit-transform-style: preserve-3d;
 -moz-transform-style: preserve-3d;
 -ms-transform-style: preserve-3d;
 transform-style: preserve-3d;
 animation: pieceRotate 5s;
 -moz-animation: pieceRotate 5s; /* Firefox */
 -webkit-animation: pieceRotate 5s; /* Safari and Chrome */
 -o-animation: pieceRotate 5s ; /* Opera */
}
/*走马灯动画*/
@keyframes pieceRotate
{
0% {-webkit-transform: rotateY(0deg);
  -ms-transform: rotateY(0deg);
  -o-transform: rotateY(0deg);
  transform: rotateY(0deg);}
100% {-webkit-transform: rotateY(360deg);
  -ms-transform: rotateY(360deg);
  -o-transform: rotateY(360deg);
  transform: rotateY(360deg);}
}
/* Firefox */
@-moz-keyframes pieceRotate
{
0% {-webkit-transform: rotateY(0deg);
  -ms-transform: rotateY(0deg);
  -o-transform: rotateY(0deg);
  transform: rotateY(0deg);}
100% {-webkit-transform: rotateY(360deg);
  -ms-transform: rotateY(360deg);
  -o-transform: rotateY(360deg);
  transform: rotateY(360deg);}
}
/* Safari and Chrome */
@-webkit-keyframes pieceRotate
{
0% {-webkit-transform: rotateY(0deg);
  -ms-transform: rotateY(0deg);
  -o-transform: rotateY(0deg);
  transform: rotateY(0deg);}
100% {-webkit-transform: rotateY(360deg);
  -ms-transform: rotateY(360deg);
  -o-transform: rotateY(360deg);
  transform: rotateY(360deg);}
}
/* Opera */
@-o-keyframes pieceRotate
{
0% {-webkit-transform: rotateY(0deg);
  -ms-transform: rotateY(0deg);
  -o-transform: rotateY(0deg);
  transform: rotateY(0deg);}
100% {-webkit-transform: rotateY(360deg);
  -ms-transform: rotateY(360deg);
  -o-transform: rotateY(360deg);
  transform: rotateY(360deg);}
}

这里就不放gif了~hhh是不是实现了酷炫的效果,还没结束哦~更酷炫的正方体组装

正方体,其实也不难实现,我这里就不很详细说了,你首先可以想象一个面,然后去拓展其他面如何去实现,比如我们把正方体的前面translateZ(100px)让它靠近我们100px,然后后面translateZ(-100px)让它远离我们100px,左边是先translateX(-100px再rotateY(90deg),右边就是translateX(100px)再rotateY(90deg),上面是先translateY(-100px),rotateX(90deg),下面是先translateY(100px),rotateX(90deg),只要我们写进动画,一切就大功告成。

css就为如下,以下就只放piece1,其他读者可以自己类比实现,或者看我github的源码

.piece-1 {
  background: #FF6666;
  -webkit-transform: rotateY(0deg) translateZ(173.2px);
  -ms-transform: rotateY(0deg) translateZ(173.2px);
  -o-transform: rotateY(0deg) translateZ(173.2px);
  transform: rotateY(0deg) translateZ(173.2px);
  animation: piece1Rotate 5s 5s;
  -moz-animation: piece1Rotate 5s 5s; /* Firefox */
  -webkit-animation: piece1Rotate 5s 5s; /* Safari and Chrome */
  -o-animation: piece1Rotate 5s 5s; /* Opera */
  -webkit-animation-fill-mode: forwards; /* Chrome, Safari, Opera */
  animation-fill-mode: forwards;
 }
/*front*/
 @keyframes piece1Rotate
 {
 0% {-webkit-transform: rotateY(0deg) translateZ(173.2px);
  -ms-transform: rotateY(0deg) translateZ(173.2px);
  -o-transform: rotateY(0deg) translateZ(173.2px);
  transform: rotateY(0deg) translateZ(173.2px);}
 100% {-webkit-transform: rotateY(0deg) translateZ(100px);
  -ms-transform: rotateY(0deg) translateZ(100px);
  -o-transform: rotateY(0deg) translateZ(100px);
  transform: rotateY(0deg) translateZ(100px);}
 }
 /* Firefox */
 @-moz-keyframes piece1Rotate
 {
 0% {-webkit-transform: rotateY(0deg) translateZ(173.2px);
  -ms-transform: rotateY(0deg) translateZ(173.2px);
  -o-transform: rotateY(0deg) translateZ(173.2px);
  transform: rotateY(0deg) translateZ(173.2px);}
 100% {-webkit-transform: rotateY(0deg) translateZ(100px);
  -ms-transform: rotateY(0deg) translateZ(100px);
  -o-transform: rotateY(0deg) translateZ(100px);
  transform: rotateY(0deg) translateZ(100px);}
 }
 /* Safari and Chrome */
 @-webkit-keyframes piece1Rotate
 {
 0% {-webkit-transform: rotateY(0deg) translateZ(173.2px);
  -ms-transform: rotateY(0deg) translateZ(173.2px);
  -o-transform: rotateY(0deg) translateZ(173.2px);
  transform: rotateY(0deg) translateZ(173.2px);}
 100% {-webkit-transform: rotateY(0deg) translateZ(100px);
  -ms-transform: rotateY(0deg) translateZ(100px);
  -o-transform: rotateY(0deg) translateZ(100px);
  transform: rotateY(0deg) translateZ(100px);}
 }
 /* Opera */
 @-o-keyframes piece1Rotate
 {
 0% {-webkit-transform: rotateY(0deg) translateZ(173.2px);
  -ms-transform: rotateY(0deg) translateZ(173.2px);
  -o-transform: rotateY(0deg) translateZ(173.2px);
  transform: rotateY(0deg) translateZ(173.2px);}
 100% {-webkit-transform: rotateY(0deg) translateZ(100px);
  -ms-transform: rotateY(0deg) translateZ(100px);
  -o-transform: rotateY(0deg) translateZ(100px);
  transform: rotateY(0deg) translateZ(100px);}
 }

细心的读者可以发现我用了一个animation-fill-mode: forwards;,这个其实就是让这些piece保持动画最后的效果,也就是正方体的效果,读者可以不加试试看,那还是会恢复原样。

最后就是正方体的旋转了,前面我们的容器已经用过animation了,读者可能会想我再加个class放animaiton不就行了,hhh,animaiton会覆盖掉前面的,那前面的走马灯的动画就没了,所以在html结构中,我再加了一个box包裹piece, 如下

<div class="container">
 <div class="piece-box">
  <div class="piece-box2"><!--新加的容器-->
   <div class="piece piece-1"></div>
   <div class="piece piece-2"></div>
   <div class="piece piece-3"></div>
   <div class="piece piece-4"></div>
   <div class="piece piece-5"></div>
   <div class="piece piece-6"></div>
  </div>
 </div>
</div>

在动画上我们可以控制正方体动画的延时时间,就是等到正方体组装完成后再开始动画

animation: boxRotate 5s 10s infinite;第一个5s是动画时长,第二个10s是延时时间,然后我们让正方体的旋转,绕X轴从0度到360度,绕Y轴也0到Y轴360度。

.piece-box2 {
 -webkit-transform-style: preserve-3d;
 -moz-transform-style: preserve-3d;
 -ms-transform-style: preserve-3d;
 transform-style: preserve-3d;
 animation: boxRotate 5s 10s infinite;
 -moz-animation: boxRotate 5s 10s infinite; /* Firefox */
 -webkit-animation: boxRotate 5s 10s infinite; /* Safari and Chrome */
 -o-animation: boxRotate 5s 10s infinite; /* Opera */
}
/*正方体旋转动画*/
@keyframes boxRotate
{
0% {-webkit-transform: rotateX(0deg) rotateY(0deg););
 -ms-transform: rotateX(0deg) rotateY(0deg););
 -o-transform: rotateX(0deg) rotateY(0deg););
 transform: rotateX(0deg) rotateY(0deg););}
100% {-webkit-transform: rotateX(360deg) rotateY(360deg);
 -ms-transform: rotateX(360deg) rotateY(360deg);
 -o-transform: rotateX(360deg) rotateY(360deg);
 transform: rotateX(360deg) rotateY(360deg);}
}
/* Firefox */
@-moz-keyframes boxRotate
{
0% {-webkit-transform: rotateX(0deg) rotateY(0deg););
 -ms-transform: rotateX(0deg) rotateY(0deg););
 -o-transform: rotateX(0deg) rotateY(0deg););
 transform: rotateX(0deg) rotateY(0deg););}
100% {-webkit-transform: rotateX(360deg) rotateY(360deg);
 -ms-transform: rotateX(360deg) rotateY(360deg);
 -o-transform: rotateX(360deg) rotateY(360deg);
 transform: rotateX(360deg) rotateY(360deg);}
}
/* Safari and Chrome */
@-webkit-keyframes boxRotate
{
0% {-webkit-transform: rotateX(0deg) rotateY(0deg););
 -ms-transform: rotateX(0deg) rotateY(0deg););
 -o-transform: rotateX(0deg) rotateY(0deg););
 transform: rotateX(0deg) rotateY(0deg););}
100% {-webkit-transform: rotateX(360deg) rotateY(360deg);
 -ms-transform: rotateX(360deg) rotateY(360deg);
 -o-transform: rotateX(360deg) rotateY(360deg);
 transform: rotateX(360deg) rotateY(360deg);}
}
/* Opera */
@-o-keyframes boxRotate
{
0% {-webkit-transform: rotateX(0deg) rotateY(0deg););
 -ms-transform: rotateX(0deg) rotateY(0deg););
 -o-transform: rotateX(0deg) rotateY(0deg););
 transform: rotateX(0deg) rotateY(0deg););}
100% {-webkit-transform: rotateX(360deg) rotateY(360deg);
 -ms-transform: rotateX(360deg) rotateY(360deg);
 -o-transform: rotateX(360deg) rotateY(360deg);
 transform: rotateX(360deg) rotateY(360deg);}
}

最后效果,大功告成!

以上就是对CSS 3D的实现例子,有需要的同学可以参考下,谢谢大家对本站的支持!

Javascript 相关文章推荐
$()JS小技巧
Jul 21 Javascript
jquery入门—选择器实现隔行变色实例代码
Jan 04 Javascript
javascript相关事件的几个概念
May 21 Javascript
基于javascript代码检测访问网页的浏览器呈现引擎、平台、Windows操作系统、移动设备和游戏系统
Dec 03 Javascript
如何利用AngularJS打造一款简单Web应用
Dec 05 Javascript
jQuery中设置form表单中action值的实现方法
May 25 Javascript
基于JavaScript实现添加到购物车效果附源码下载
Aug 22 Javascript
强大Vue.js组件浅析
Sep 12 Javascript
smartupload实现文件上传时获取表单数据(推荐)
Dec 12 Javascript
AngularJs分页插件使用详解
Jun 30 Javascript
深入浅出理解JavaScript闭包的功能与用法
Aug 01 Javascript
jQuery使用jsonp实现百度搜索的示例代码
Jul 08 jQuery
js实现StringBuffer的简单实例
Sep 02 #Javascript
纯JS实现可拖拽表单的简单实例
Sep 02 #Javascript
vue.js入门教程之绑定class和style样式
Sep 02 #Javascript
js绘制购物车抛物线动画
Nov 18 #Javascript
基于jQuery实现发送短信验证码后的倒计时功能(无视页面关闭)
Sep 02 #Javascript
基于JS实现发送短信验证码后的倒计时功能(无视页面刷新,页面关闭不进行倒计时功能)
Sep 02 #Javascript
node.js中module.exports与exports用法上的区别
Sep 02 #Javascript
You might like
php下实现农历日历的代码
2007/03/07 PHP
joomla数据库操作示例代码
2016/01/06 PHP
thinkphp关于简单的权限判定方法
2017/04/03 PHP
php获取ajax的headers方法与内容实例
2017/12/27 PHP
jquery键盘事件使用介绍
2011/11/01 Javascript
JQuery中阻止事件冒泡几种方式及其区别介绍
2014/01/15 Javascript
JQuery设置获取下拉菜单某个选项的值(比较全)
2014/08/05 Javascript
第二章之Bootstrap 页面排版样式
2016/04/25 Javascript
jQuery实现点击弹出背景变暗遮罩效果实例代码
2016/06/24 Javascript
JS实现淡入淡出图片效果的方法分析
2016/12/20 Javascript
Javascript的this用法
2017/01/16 Javascript
jQuery插件HighCharts绘制2D金字塔图效果示例【附demo源码下载】
2017/03/09 Javascript
ES6教程之for循环和Map,Set用法分析
2017/04/10 Javascript
JQuery实现定时刷新功能代码
2017/05/09 jQuery
在vue中使用vue-echarts-v3的实例代码
2018/09/13 Javascript
JS 验证码功能的三种实现方式
2018/11/26 Javascript
详解将微信小程序接口Promise化并使用async函数
2019/08/05 Javascript
vue-preview动态获取图片宽高并增加旋转功能的实现
2020/07/29 Javascript
原生js实现下拉框选择组件
2021/01/20 Javascript
[02:52]DOTA2新手基础教程 米波
2014/01/21 DOTA
[02:31]2018年度DOTA2最具人气选手-完美盛典
2018/12/16 DOTA
elasticsearch python 查询的两种方法
2019/08/04 Python
twilio python自动拨打电话,播放自定义mp3音频的方法
2019/08/08 Python
解决Tensorflow 使用时cpu编译不支持警告的问题
2020/02/03 Python
关于windows下Tensorflow和pytorch安装教程
2020/02/04 Python
Opencv+Python识别PCB板图片的步骤
2021/01/07 Python
浅析python连接数据库的重要事项
2021/02/22 Python
W Concept美国:精选全球独立设计师
2017/02/22 全球购物
资生堂英国官网:Shiseido英国
2020/12/30 全球购物
汉语言文学毕业生求职信
2013/10/01 职场文书
《云雀的心愿》教学反思
2014/02/25 职场文书
酒店周年庆活动方案
2014/08/21 职场文书
私用公车造成事故检讨书
2014/11/16 职场文书
2016年大学自主招生自荐信范文
2015/03/24 职场文书
信息技术课教学反思
2016/02/23 职场文书
读完《骆驼祥子》的观后感!
2019/07/05 职场文书