侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

详解CSS3的perspective属性设置3D变换距离的方法

前端  /  管理员 发布于 7年前   245

perspective属性对于3D变形来说至关重要。该属性会设置查看者的位置,并将可视内容映射到一个视锥上,继而投到一个2D视平面上。如果不指定透视,则Z轴空间中的所有点将平铺到同一个2D视平面中,并且变换结果中将不存在景深概念。

上面的描述可能让人难以理解一些,其实对于perspective属性,我们可以简单的理解为视距,用来设置用户和元素3D空间Z平面之间的距离。而其效应由他的值来决定,值越小,用户与3D空间Z平面距离越近,视觉效果更令人印象深刻;反之,值越大,用户与3D空间Z平面距离越远,视觉效果就很小。

在3D变形中,对于某些变形,例如下面的示例演示的沿Z轴的变形,perspective属性对于查看变形的效果来说必不可少。

我们先来看一个简单的实例,制作一个扑克牌3D旋转效果,并且一个在扑克牌的父元素添加了视距perspective,而另一个却没有设置:

HTML

XML/HTML Code复制内容到剪贴板
  1. <div>  
  2.     <img src="images/cardKingClub.png" alt="" width="142" height="200" />  
  3.     <img src="images/cardKingClub.png" alt="" width="142" height="200" />  
  4. </div>  
  5. <div>  
  6.     <img src="images/cardKingClub.png" alt="" width="142" height="200" />  
  7.     <img src="images/cardKingClub.png" alt="" width="142" height="200" />  
  8. </div>  

CSS

CSS Code复制内容到剪贴板
  1. div {   
  2.     width: 500px;   
  3.     height: 300px;   
  4.     margin: 30px auto;   
  5.     position: relative;   
  6.     background: url(images/bg-grid.jpg) no-repeat center center;   
  7.     background-size: 100% 100%;   
  8. }   
  9. div img {   
  10.     position: absolute;   
  11.     top: 50%;   
  12.     left: 50%;   
  13.     margin-left: -71px;   
  14.     margin-top: -100px;    
  15.     transform-origin: bottombottom;   
  16. }   
  17. div img:nth-child(1){   
  18.     opacity: .5;   
  19.     z-index: 1;   
  20. }   
  21. div img:nth-child(2){   
  22.     z-index: 2;   
  23.     transform: rotateX(45deg);   
  24. }   
  25. div:nth-of-type(2){   
  26.     perspective: 500px;   
  27. }  

其效果如下:

上图的效果完全说明了一切。父节点没有设置perspective的情况下,梅花King的3D旋转效果不明显,而在父节点设置perspective后,梅花King才像个3D旋转。

上例简单的演示了perspective的使用方法,我们回过头来,看看perspective的使用语法:

CSS Code复制内容到剪贴板
  1. perspective:none | <length>  

perspective属性包括两个属性:none和具有单位的长度值。其中perspective属性的默认值为none,表示无限的角度来看3D物体,但看上去是平的。另一个值<length>接受一个长度单位大于0的值。而且其单位不能为百分比值。<length>值越大,角度出现的越远,从而创建一个相当低的强度和非常小的3D空间变化。反之,此值越小,角度出现的越近,从而创建一个高强度的角度和一个大型3D变化。

比如你站在10英尺和1000英尺的地方看一个10英尺的立方体。在10英尺的地方,你距离立方体是一样的尺寸。因此视角转变远远大于站在1000英尺处的,立体尺寸是你距离立方体距离的百分之一。同样的思维适用于perspective的<length>值。我们一起来看一个实例,来加强这方面的理解:

HTML

XML/HTML Code复制内容到剪贴板
  1. <div class="wrapper w2">  
  2.     <div class="cube">  
  3.         <div class="side  front">1</div>  
  4.         <div class="side   back">6</div>  
  5.         <div class="side  right">4</div>  
  6.         <div class="side   left">3</div>  
  7.         <div class="side    top">5</div>  
  8.         <div class="side bottom">2</div>  
  9.     </div>  
  10. </div>  
  11. <div class="wrapper w1">  
  12.     <div class="cube">  
  13.         <div class="side  front">1</div>  
  14.         <div class="side   back">6</div>  
  15.         <div class="side  right">4</div>  
  16.         <div class="side   left">3</div>  
  17.         <div class="side    top">5</div>  
  18.         <div class="side bottom">2</div>  
  19.     </div>  
  20. </div>  

CSS

CSS Code复制内容到剪贴板
  1. .wrapper {   
  2.     width: 50%;   
  3.     float: left;   
  4. }   
  5. .cube {   
  6.     font-size: 4em;   
  7.     width: 2em;   
  8.     margin: 1.5em auto;   
  9.     transform-style: preserve-3d;   
  10.     transform: rotateX(-40deg) rotateY(32deg);   
  11. }   
  12. .side {   
  13.     position: absolute;   
  14.     width: 2em;   
  15.     height: 2em;   
  16.     background: rgba(255, 99, 71, 0.6);   
  17.     border: 1px solid rgba(0, 0, 0, 0.5);   
  18.     color: white;   
  19.     text-align: center;   
  20.     line-height: 2em;   
  21. }   
  22. .front {   
  23.     transform: translateZ(1em);   
  24. }   
  25. .top {   
  26.     transform: rotateX(90deg) translateZ(1em);   
  27. }   
  28. .rightright {   
  29.     transform: rotateY(90deg) translateZ(1em);   
  30. }   
  31. .left {   
  32.     transform: rotateY(-90deg) translateZ(1em);   
  33. }   
  34. .bottombottom {   
  35.     transform: rotateX(-90deg) translateZ(1em);   
  36. }   
  37.   
  38. .back {   
  39.     transform: rotateY(-180deg) translateZ(1em);   
  40. }   
  41. .w1 {   
  42.     perspective: 100px;   
  43. }   
  44. .w2{   
  45.     perspective: 1000px;   
  46. }  

效果如下图所示:

依据上面的介绍,我们可对perspective取值做一个简单的结论:

 1.perspective取值为none或不设置,就没有真3D空间。
 2.perspective取值越小,3D效果就越明显,也就是你的眼睛越靠近真3D。
 3.perspective的值无穷大,或值为0时与取值为none效果一样。
为了更好的理解perspective属性,我们很有必要把他和translateZ的关系结合起来。其实也可以把perspective的值简单的理解为人的眼睛到显示器的距离,而translate就是3D物体距离源点的距离,下面引用W3C的一张图来解说perspective和translateZ的关系。

上图显示了perspective属性和translateZ的位置比例。要顶部的图,Z是半个d,为了使用原始圆(轮廓)看起来出现在Z轴上(虚线圆),画布上的实体圆将扩大两部,如浅蓝色的圆。在底部图中显示,圆按比例缩小,致使虚线圆出现在画布后面,并且z的大小是距原始位置三分之一。

在3D变形中,除了perspective属性可以激活一个3D空间之外,在3D变形的函数中的perspective()也可以激活3D空间。他们不同的地方是:perspective用在舞台元素上(变形元素们的共同父元素);perspective()就是用在当前变形元素上,并且可以与其他的transform函数一起使用。例如,我们可以把:

CSS Code复制内容到剪贴板
  1. .stage {   
  2.     perspective: 600px  
  3. }  

写成:

CSS Code复制内容到剪贴板
  1. .stage .box {   
  2.     transform: perspective(600px);   
  3. }  

来看一个简单示例:

HTML

XML/HTML Code复制内容到剪贴板
  1. <div class="stage">  
  2.     <div class="container">  
  3.         <img src="images/cardKingClub.png" alt="" width="142" height="200" />  
  4.     </div>  
  5. </div>  
  6. <div class="stage">  
  7.     <div class="container">  
  8.         <img src="images/cardKingClub.png" alt="" width="142" height="200" />  
  9.     </div>  
  10. </div>  

CSS

CSS Code复制内容到剪贴板
  1. .stage {   
  2.     width: 500px;   
  3.     height: 300px;   
  4.     margin: 30px auto;   
  5.     position: relative;   
  6.     background: url(images/bg-grid.jpg) no-repeat center center;   
  7.     background-size: 100% 100%;   
  8. }   
  9. .container {   
  10.     position: absolute;   
  11.     top: 50%;   
  12.     left: 50%;   
  13.     width: 142px;   
  14.     height: 200px;   
  15.     border: 1px dotted orange;   
  16.     margin-left: -71px;   
  17.     margin-top: -100px;    
  18. }   
  19. .container img{   
  20.     transform: rotateY(45deg);   
  21. }   
  22. .stage:nth-child(1) .container{   
  23.     perspective: 600px;   
  24. }   
  25. .stage:nth-child(2) img {   
  26.     transform:perspective(600px) rotateY(45deg);   
  27. }  

效果如下所示:

上图效果可以看出,虽然书写的形式,属性名称不一致,但是效果却一样。

虽然perspective属性和perspective()函数所起的功能是一样的,但其取值以及以运用的对像有所不同:

1. perspective属性可以取值为none或长度值;而perspective()函数取值只能大于0,如果取值为0或比0小的值,将无法激活3D空间;
2.perspective属性用于变形对像父元素;而perspective()函数用于变形对像自身,并和transform其他函数一起使用。
 

perspective-origin属性

perspective-origin属性是3D变形中另一个重要属性,主要用来决定perspective属性的源点角度。它实际上设置了X轴和Y轴位置,在该位置观看者好像在观看该元素的子元素。

perspective-origin属性的使用语法也很简单:


复制代码代码如下:perspective-origin:[<percentage> | <length> | left | center | right | top | bottom] | [[<percentage> | <length> | left | center | right] && [<percentage> | <length> | top | center | bottom]]
该属性的默认值为“50% 50%”(也就是center),其也可以设置为一个值,也可以设置为两个长度值:

 第一个长度值指定相对于元素的包含框的X轴上的位置。它可以是长度值(以受支持的长度单位表示)、百分比或以下三个关键词之一:left(表示在包含框的X轴方向长度的0%),center(表示中间点),或right(表示长度的100%)。
 第二个长度值指定相对于元素的包含框的Y轴上的位置。它可以是长度值、百分比或以下三个关键词之一:top(表示在包含框的Y轴方向长度的0%),center(表示中间点),或bottom(表示长度的100%)。
注意,为了指转换子元素变形的深度,perspective-origin属性必须定义父元素上。通常perspective-origin属性本身不做任何事情,它必须被定义在设置了perspective属性的元素上。换句话说,perspective-origin属性需要与perspective属性结合起来使用,以便将视点移至元素的中心以外位置,如下图所示:

往往我们看一样东西不可能一直都在中心位置看,想换个角度,换个位置一看究竟,这个时候就离不开perspective-origin这个属性,下面来自于W3C官网的图可以简单阐述这一观点:


  • 上一条:
    记住CSS中的10个不要
    下一条:
    CSS3中设置3D变形的transform-style属性详解
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 使用 Alpine.js 排序插件对元素进行排序(0个评论)
    • 在js中使用jszip + file-saver实现批量下载OSS文件功能示例(0个评论)
    • 在vue中实现父页面按钮显示子组件中的el-dialog效果(0个评论)
    • 使用mock-server实现模拟接口对接流程步骤(0个评论)
    • vue项目打包程序实现把项目打包成一个exe可执行程序(0个评论)
    • 近期文章
    • 智能合约Solidity学习CryptoZombie第三课:组建僵尸军队(高级Solidity理论)(0个评论)
    • 智能合约Solidity学习CryptoZombie第二课:让你的僵尸猎食(0个评论)
    • 智能合约Solidity学习CryptoZombie第一课:生成一只你的僵尸(0个评论)
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-06
    • 2017-07
    • 2017-08
    • 2017-09
    • 2017-10
    • 2017-11
    • 2018-03
    • 2018-04
    • 2018-05
    • 2018-06
    • 2018-09
    • 2018-11
    • 2018-12
    • 2019-02
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2021-04
    • 2021-05
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-11
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-09
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-02
    • 2024-03
    • 2024-04
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客