灵活的椭圆
这里要注意,border-radius是个简写,它可以拆成4个部分,8个值。
border-top-left-radius border-top-right-radius border-bottom-right-radius border-bottom-left-radius
可以简写到一个里面,规则和padding什么的一样。
border-radius: 50% / 50px 20px;
比如这个,就相当于:
border-bottom-left-radius: 50% 20px; border-bottom-right-radius: 50% 50px; border-top-left-radius: 50% 50px; border-top-right-radius: 50% 20px;
那半个椭圆就是:
border-radius: 50% / 100% 100% 0 0;
4分之一个椭圆:
border-radius: 100% 0 0 0;
平行四边形
这里准备采用的办法是使用transform来水平拉伸矩形成为平行四边形,如果直接这样做会同时把里面的内容拉伸,解决办法就是使用一个伪元素。(如果你想使用两个元素的话。。。当我啥都没说)
#Parallelograms { position: relative; padding:10px; margin:10px; }#Parallelograms::before { content: ''; /* To generate the box */ position: absolute; top: 0; right: 0; bottom: 0; left: 0; z-index: -1; background: #58a; transform: skew(-45deg); }
这里的伪元素相对生成元素使用绝对定位,并完全规定了所有位置将伪元素拉满了生成元素。然后把它扯歪。这样就不会影响到元素内的内容了。
这种做法不止适用于菱形,还适用于所有想要变形但不影响内容的操作。还可以用来在IE8中产生多重背景,之前的外方内圆也可以用这个来实现。或者使用这个来创建使用opacity属性只透明背景色。或者多重边框也可以。
菱形图片
之前一般使用图片处理软件来实现,但是这样显然不够灵活。
使用变形
<div class="picture"> <img src="img/adamcatlace.jpg" alt="..." /></div>.picture { margin:50px; width: 200px; transform: rotate(45deg); overflow: hidden; } .picture > img { max-width: 100%; transform: rotate(-45deg) scale(1.42); }
这里之所以使用scale来放大img而不直接使用width,是因为在浏览器不支持transform时显示还是正常的,否则图片会被直接放大,现在这样只有支持transform的浏览器会对图片进行旋转和放大。
这个方法并不优雅,使用了新的HTML元素。如果图片不是正方的。。。那就悲剧了。。
clip-path
这是从SVG那边借鉴来的特性直接应用在元素上,把元素剪为指定的形状。
在这里使用的是多边形polygon()函数,这个函数通过规定多边形端点在元素原本矩形形状中的位置来创建。
我们之前要创建的形状这里就是polygon(50% 0, 100% 50%, 50% 100%, 0 50%);
这里使用这个还有一个很棒的好处,这个属性是直接支持CSS动画的,只要函数相同(同是polygon),顶点数也相同,就可以应用动画。
#picture-clip-path { -webkit-clip-path: polygon(50% 0, 100% 50%, 50% 100%, 0 50%); clip-path: polygon(50% 0, 100% 50%,50% 100%, 0 50%); transition: 1s; }#picture-clip-path:hover { clip-path: polygon(0 0, 100% 0,100% 100%, 0 100%); -webkit-clip-path: polygon(0 0, 100% 0,100% 100%, 0 100%); }
切角
就是元素有一个或多个角是不全的,这个以前多是由图片或hack(添加一个绝对定位元素)实现的。显然应该有更好的方法。
gradient
这个只使用有方向的linear gradient就可以解决。
#corner-cut1{ width:100px; height:100px; background: #58a; background: linear-gradient(135deg, transparent 15px, #58a 0) top left, linear-gradient(-135deg, transparent 15px, #655 0) top right, linear-gradient(-45deg, transparent 15px, #58a 0) bottom right, linear-gradient(45deg, transparent 15px, #655 0) bottom left; background-size: 50% 50%; background-repeat: no-repeat; }
4个角,对应4个背景,每个背景占1/4,每个背景都是一个相应方向的linear gradient。要注意,linear gradient这个东西是默认接管整个元素背景的,如果不禁止重复的话,会造成最后一个覆盖所有。
这个如果使用SCSS的mixin会方便一点:
@mixin beveled-corners($bg, $tl:0, $tr:$tl, $br:$tl, $bl:$tr) { background: $bg; background: linear-gradient(135deg, transparent $tl, $bg 0) top left, linear-gradient(225deg, transparent $tr, $bg 0) top right, linear-gradient(-45deg, transparent $br, $bg 0) bottom right, linear-gradient(45deg, transparent $bl, $bg 0) bottom left; background-size: 50% 50%; background-repeat: no-repeat; }#corner-cut1{ width:100px; height:100px; @include beveled-corners(#58a, 15px, 5px);}
这里使用了默认值,传1-5个参数都可以。
弯曲的切角
这里的实现和上面的思想是一样的,换成radial-gradient即可:
#corner-cut2{ width:100px; height:100px; background: #58a; background: radial-gradient(circle at top left, transparent 15px, #58a 0) top left, radial-gradient(circle at top right, transparent 15px, #58a 0) top right, radial-gradient(circle at bottom right, transparent 15px, #58a 0) bottom right, radial-gradient(circle at bottom left, transparent 15px, #58a 0) bottom left; background-size: 50% 50%; background-repeat: no-repeat; }
SVG加border-image
使用上面的办法有几个问题,首先就是代码太复杂,要改颜色和切角的大小要改的太多了;还有就是这种方式并不能支持动画。
我们还可以采用外的两个方法来克服这个缺点,这个方法需要写内联的SVG代码。
#corner-cut3{ border: 20px solid #58a; border-image: 1 url('data:image/svg+xml,\ <svg xmlns="http://www.w3.org/2000/svg" width="3" height="3" fill="%2358a">\ <polygon points="0,1 1,0 2,0 3,1 3,2 2,3 1,3 0,2"/>\ </svg>'); background: #58a; background-clip: padding-box; }
这里要控制切角的大小,只需改变border的宽度,这个宽度是可以应用CSS动画的哦。而且背景与边框的样式不再关联了,背景想是啥是啥。这里给border设置了一个颜色作为fallback。
clip-path
以上两种方法的局限是只能使用很有限样式的边框,如果我们想要图片是背景而且要切角呢?
就用之前的办法咯~
#corner-cut4{ background-image: url("../img/adamcatlace.jpg"); width:100px; height:100px; -webkit-clip-path: polygon(20px 0, calc(100% - 20px) 0, 100% 20px, 100% calc(100% - 20px), calc(100% - 20px) 100%, 20px 100%, 0 calc(100% - 20px), 0 20px ); clip-path: polygon(20px 0, calc(100% - 20px) 0, 100% 20px, 100% calc(100% - 20px), calc(100% - 20px) 100%, 20px 100%, 0 calc(100% - 20px), 0 20px ); }
将来
在Level 4的CSS背景与边框中,新引入了corner-shape属性,将来就方便咯。
梯形
梯形的形状广泛的被用于标签页的形状。
这里利用transform 3D,使用近大远小的效果创建两边倾斜的元素。
.trapezoid { position: relative; display: inline-block; padding: .3em 1em 0; margin-right:-15px; //background: rgba(255,255,255,0.5);} .trapezoid::before { content: ''; position: absolute; top: 0; right: 0; bottom: 0; left: 0; z-index: -1; background: #ccc; background-image: linear-gradient(hsla(0,0%,100%,.6), hsla(0,0%,100%,0)); border: 1px solid rgba(0,0,0,.4); border-bottom: none; border-radius: .5em .5em 0 0; box-shadow: 0 .15em white inset; transform: perspective(.5em) rotateX(5deg); transform-origin: bottom; }
相较于以前前后加两个伪元素使用border来模拟,这个方法更加灵活,想给整个梯形加个边框加个圆角什么的非常容易,原来的方法你试试。
通过改变transform-origin,可以改变梯形两边倾斜的角度,或者只让一边倾斜。
简单的饼图
使用transform
我们想只用一个元素,最多再加一个伪元素来实现这个。
我们首先需要一个圆:
.pie { width: 100px; height: 100px; border-radius: 50%; background: yellowgreen; }
我们把它分为两半,左边的一半颜色是背景,右边半的颜色用来表示百分比:
background-image: linear-gradient(to right, transparent 50%, #655 0);
此时这个元素的样子就像是50%的饼图。
接下来就轮到伪元素上场了,在0-50%时,我们使用一个与背景色相同的半圆伪元素,利用transform围绕圆的中心旋转遮住元素右边的一部分,使用未遮住的部分来表示百分比。在50%-100%时,使用一个与百分比背景颜色相同的半圆伪元素,遮住部分左边背景色部分,并与原元素右边一起组成大于50%的百分比。
.pie::before { content: ''; display: block; margin-left: 50%; height: 100%; border-radius: 0 100% 100% 0 / 50%; background: #655; transform-origin: left; transform: rotate(.1turn); }
好了,这样的话我们有了一个可以使用transform:rotate和伪元素背景来控制的饼图。这显然是不灵活的。
不过想要直接通过这样的方式来控制CSS显然是不现实的:
<div class="pie">20%</div> <div class="pie">60%</div>
为了实现这样的灵活性,我们首先做一个铺垫,使用CSS3动画循环整个0-100%的过程:
@keyframes spin { to { transform: rotate(.5turn); } } @keyframes bg { 50% { background: #655; } }.pie { position: relative; width: 100px; line-height: 100px; border-radius: 50%; background: yellowgreen; background-image: linear-gradient(to right, transparent 50%, #655 0); color: transparent; text-align: center; }.pie::before { content: ''; display: block; margin-left: 50%; height: 100%; border-radius: 0 100% 100% 0 / 50%; background-color: inherit; transform-origin: left; animation: spin 3s linear infinite, bg 6s step-end infinite; }
这里的动画在6秒钟的过程中,前3秒伪元素背景就是背景的颜色,转180度到50%;
在第3秒时,伪元素背景变为百分比的颜色并回到初始位置,与刚才结束时一样表示50%,但是在这个瞬间是有闪烁的,不过没关系,我们最后并不真的使用这个动画效果本身。
后三秒伪元素转180度,结束在100%的位置。
好了,那么这个动画到底有什么作用呢?
我们使用一个我们不太好想到的特性,将动画的animation-delay设为一个负值,这个属性本来是控制动画延迟开始时间的,当它是负值时,规范规定它和将这个值设为0一样,动画会马上开始,但是动画会自动的从这个负值的绝对值处开始,也就是这个值前面的那些动画被跳过。想想这个设定挺符合逻辑的。
再使用animation-play-state: paused使动画一开始就停住。
animation: spin 50s linear infinite, bg 100s step-end infinite; animation-play-state: paused; animation-delay: inherit;
这样我们就可以直接使用内联样式来控制饼图了(记得把伪元素的animation-delay设置为继承):
<div class="pie" ></div> <div class="pie" ></div>
如果你一定要用:
<div class="pie">20%</div> <div class="pie">60%</div>
再使用JS将这个值读出来赋给style就好。
这个思想适用于很多地方,比如你想要个渐变的颜色,你又懒得自己调颜色,那你可以使用这种办法,一个从一种颜色到另一种的动画。
.pie { position: relative; width: 100px; line-height: 100px; border-radius: 50%; background: yellowgreen; background-image: linear-gradient(to right, transparent 50%, #655 0); color: transparent; text-align: center; } @keyframes spin { to { transform: rotate(.5turn); } } @keyframes bg { 50% { background: #655; } }.pie::before { content: ''; position: absolute; top: 0; left: 50%; width: 50%; height: 100%; border-radius: 0 100% 100% 0 / 50%; background-color: inherit; transform-origin: left; animation: spin 50s linear infinite, bg 100s step-end infinite; animation-play-state: paused; animation-delay: inherit; }
SVG
利用一个圆,加一个dash形状的粗边框来实现,dash的粗边框可以控制dash的长度和空的长度,设置为只有一个dash,并通过控制这个dash的宽度来表示百分比。
<svg viewBox="0 0 32 32"> <circle r="16" cx="16" cy="16" /> </svg>.piesvg svg { width: 100px; height: 100px; transform: rotate(-90deg); background: yellowgreen; border-radius: 50%; } .piesvg circle { fill: yellowgreen; stroke: #655; stroke-width: 32; stroke-dasharray: 38 100; /* for 38% */ }
我们可以直接使用js来创建svg
.piesvg svg { width: 100px; height: 100px; transform: rotate(-90deg); background: yellowgreen; border-radius: 50%; } .piesvg circle { fill: yellowgreen; stroke: #655; stroke-width: 32; } $$('.piesvg').forEach(function(pie) { var p = parseFloat(pie.textContent); var NS = "http://www.w3.org/2000/svg"; var svg = document.createElementNS(NS, "svg"); var circle = document.createElementNS(NS, "circle"); var title = document.createElementNS(NS, "title"); circle.setAttribute("r", 16); circle.setAttribute("cx", 16); circle.setAttribute("cy", 16); circle.setAttribute("stroke-dasharray", p + " 100"); svg.setAttribute("viewBox", "0 0 32 32"); title.textContent = pie.textContent; pie.textContent = ''; svg.appendChild(title); svg.appendChild(circle); pie.appendChild(svg); }); <div class="piesvg">25%</div> <div class="piesvg">35%</div>
如果你想使用动画的话,直接对这个属性动画就好了:stroke-dasharray。
作者:exialym
链接:https://www.jianshu.com/p/e7fa44a6cc17