神奇的 conic-gradient 圆锥渐变

dkol9334 7年前
   <p>conic-gradient 是个什么?说到  conic-gradient ,就不得不提的它的另外两个兄弟:</p>    <ul>     <li>linear-gradient  : 线性渐变</li>     <li>radial-gradient  : 径向渐变<br> <img src="https://simg.open-open.com/show/a735d84e934e0401f964175e8f963ecb.png"></li>    </ul>    <p>说这两个应该还是有很多人了解并且使用过的。CSS3 新增的线性渐变及径向渐变给 CSS 世界带来了很大的变化。</p>    <p>而  conic-gradient ,表示圆锥渐变,另外一种渐变方式,给 CSS 世界带来了更多可能。</p>    <p>下面进入正文,本文中所有示例,请在高版本 Chrome 内核下预览。</p>    <h2>API</h2>    <p>看看它最简单的 API:</p>    <pre>  <code class="language-css">{      /* Basic example */       background: conic-gradient(deeppink, yellowgreen);  }</code></pre>    <p><img src="https://simg.open-open.com/show/cb366454a0c9526677274f636287d52d.png"></p>    <h3>与线性渐变及圆锥渐变的异同</h3>    <p>那么它和另外两个渐变的区别在哪里呢?</p>    <ul>     <li>linear-gradient  线性渐变的方向是一条直线,可以是任何角度</li>     <li>radial-gradient 径向渐变是从圆心点以椭圆形状向外扩散</li>    </ul>    <p>而从方向上来说,圆锥渐变的方向是这样的:</p>    <p><img src="https://simg.open-open.com/show/46651bb37c8aa28234325364c8b2808b.gif"></p>    <p>划重点:</p>    <p>从图中可以看到,圆锥渐变的渐变方向和起始点。 起始点是图形中心,然后以顺时针方向绕中心实现渐变效果 。</p>    <h2>使用  conic-gradient 实现颜色表盘</h2>    <p>从上面了解了  conic-gradient 最简单的用法,我们使用它实现一个最简单的颜色表盘。</p>    <p>conic-gradient 不仅仅只是从一种颜色渐变到另一种颜色,与另外两个渐变一样,可以实现多颜色的过渡渐变。</p>    <p>由此,想到了彩虹,我们可以依次列出  赤橙黄绿青蓝紫 七种颜色:</p>    <ul>     <li>conic-gradient: (red, orange, yellow, green, teal, blue, purple)</li>    </ul>    <p>上面表示,在圆锥渐变的过程中,颜色从设定的第一个  red 开始,渐变到  orange ,再到  yellow ,一直到最后设定的  purple 。并且每一个区间是等分的。</p>    <p>我们再给加上  border-radius: 50% ,假设我们的 CSS 如下,</p>    <pre>  <code class="language-css">{      width: 200px;      height: 200px;      border-radius: 50%;      background: conic-gradient(red, orange, yellow, green, teal, blue, purple);  }</code></pre>    <p>看看效果:</p>    <p><img src="https://simg.open-open.com/show/a45069fd73f865c08ecdcd0e9657b79d.png"></p>    <p>wow,已经有了初步形状了。但是,总感觉哪里不大自然。总之,浑身难受  <img src="https://simg.open-open.com/show/a1e2372a8d17a340aa1addccaea8a082.jpg"></p>    <p>嗯?问题出在哪里呢?一是颜色不够丰富不够明亮,二是起始处与结尾处衔接不够自然。让我再稍微调整一下。</p>    <p>我们知道,表示颜色的方法,除了  rgb() 颜色表示法之外,还有  hsl() 表示法。</p>    <p>hsl() 被定义为色相-饱和度-明度(Hue-saturation-lightness)</p>    <ul>     <li>色相(H)是色彩的基本属性,就是平常所说的颜色名称,如红色、黄色等。</li>     <li>饱和度(S)是指色彩的纯度,越高色彩越纯,低则逐渐变灰,取0-100%的数值。</li>     <li>明度(V),亮度(L),取0-100%。</li>    </ul>    <p>这里,我们通过改变色相得到一个较为明亮完整的颜色色系。</p>    <p>也就是采用这样一个过渡  hsl(0%, 100%, 50%) -->  hsl(100%, 100%, 50%) ,中间只改变色相,生成 20 个过渡状态。借助 SCSS ,CSS 语法如下:</p>    <pre>  <code class="language-css">$colors: ();  $totalStops:20;    @for $i from 0 through $totalStops{      $colors: append($colors, hsl($i *(360deg/$totalStops), 100%, 50%), comma);  }    .colors {      width: 200px;      height: 200px;      background: conic-gradient($colors);      border-radius: 50%;  }</code></pre>    <p>得到如下效果图,这次的效果很好:</p>    <p><img src="https://simg.open-open.com/show/8d4dceaa23b5d39b6b1483d8efc8443d.png"></p>    <p><a href="/misc/goto?guid=4959750403109535241" rel="nofollow,noindex">CodePen Demo -- conic-gradinet colors</a></p>    <h2>配合百分比使用</h2>    <p>当然,我们可以更加具体的指定圆锥渐变每一段的比例,配合百分比,可以很轻松的实现饼图。</p>    <p>假设我们有如下 CSS:</p>    <pre>  <code class="language-css">{      width: 200px;      height: 200px;      background: conic-gradient(deeppink 0, deeppink 30%, yellowgreen 30%, yellowgreen 70%, teal 70%, teal 100%);      border-radius: 50%;  }</code></pre>    <p>上图,我们分别指定了 0~30%,30%~70%,70%~100% 三个区间的颜色分别为  deeppink(深红) , yellowgreen(黄绿) 以及  teal(青) ,可以得到如下饼图:</p>    <p><img src="https://simg.open-open.com/show/487545fae8c9f8a3aa6002c0c692bc9e.png"></p>    <p>当然,上面只是百分比的第一种写法,还有另一种写法也能实现:</p>    <pre>  <code class="language-css">{      background: conic-gradient(deeppink 0 30%, yellowgreen 0 70%, teal 0 100%);  }</code></pre>    <p>这里表示 :</p>    <ol>     <li>0-30% 的区间使用  deeppink</li>     <li>0-70% 的区间使用  yellowgreen</li>     <li>0-100% 的区间使用  teal</li>    </ol>    <p>而且,先定义的颜色的层叠在后定义的颜色之上。</p>    <p><a href="/misc/goto?guid=4959750403222897570" rel="nofollow,noindex">CodePen Demo -- use proportion in conic-gradient</a></p>    <h2>配合  background-size 使用</h2>    <p>使用了百分比控制了区间,再配合使用  background-size 就可以实现各种贴图啦。</p>    <p>我们首先实现一个基础圆锥渐变图形如下:</p>    <pre>  <code class="language-css">{      width: 250px;      height: 250px;      margin: 50px auto;      background: conic-gradient(#000 12.5%, #fff 0 37.5%, #000 0 62.5%, #fff 0 87.5%, #000 0);  }</code></pre>    <p>效果图:</p>    <p><img src="https://simg.open-open.com/show/6bd23b848fcc325f866926470f5e5775.png"></p>    <p>再加上  background-size: 50px 50px; ,也就是:</p>    <pre>  <code class="language-css">{      width: 250px;      height: 250px;      margin: 50px auto;      background: conic-gradient(#000 12.5%, #fff 0 37.5%, #000 0 62.5%, #fff 0 87.5%, #000 0);      background-size: 50px 50px;  }</code></pre>    <p>得到:</p>    <p><img src="https://simg.open-open.com/show/74c4d3716b3962e55a0674efd1876902.png"></p>    <p><a href="/misc/goto?guid=4959750403337524102" rel="nofollow,noindex">CodePen Demo -- conic-gradient wallpaper</a></p>    <h2>重复圆锥渐变  repaeting-conic-gradient</h2>    <p>与线性渐变及径向渐变一样,圆锥渐变也是存在重复圆锥渐变  repaet-conic-gradient 的。</p>    <p>我们假设希望不断重复的片段是 0~30° 的一个片段,它的 CSS 代码是  conic-gradient(deeppink 0 15deg, yellowgreen 0 30deg) 。</p>    <p><img src="https://simg.open-open.com/show/5947e5c3ef15759cc68f1ddec30c7acb.png"></p>    <p>那么,使用了  repaeting-conic-gradient 之后,会自动填充满整个区域,CSS 代码如下:</p>    <pre>  <code class="language-css">{      width: 200px;      height: 200px;      background: repeating-conic-gradient(deeppink 0 15deg, yellowgreen 0 30deg);      border: 1px solid #000;  }</code></pre>    <p><img src="https://simg.open-open.com/show/70543cf8e1537c1e9b5195d6947f8bbb.png"></p>    <p><a href="/misc/goto?guid=4959750403448411998" rel="nofollow,noindex">CodePen Demo -- repeating-conic-gradient</a></p>    <h2>圆锥渐变动画</h2>    <p>基本的一些用法了解完了,看看使用圆锥渐变可以玩出什么花来。</p>    <p>借助  SCSS 的强大,我们可以制作出一些非常酷炫的背景展板。</p>    <p>使用 SCSS ,我们随机生成一个多颜色的圆锥渐变图案:</p>    <p>假设我们的 HTML 结构如下:</p>    <pre>  <code class="language-css"><div></div></code></pre>    <p>CSS/SCSS 代码如下:</p>    <pre>  <code class="language-css">@function randomNum($max, $min: 0, $u: 1) {   @return ($min + random($max)) * $u;  }    @function randomConicGradient() {      $n: random(30) + 10;   $list: ();         @for $i from 0 to $n {    $list: $list, rgb(randomNum(255), randomNum(255), randomNum(255));   }       @return conic-gradient($list, nth($list, 1));  }    div {      width: 100vw;      height: 100vh;      background: randomConicGradient();  }</code></pre>    <p>简单解释下上面的 SCSS 代码,</p>    <ul>     <li>randomNum()  用于生成随机数, randomNum(255)  相当于随机生成 1~255 的随机数;</li>     <li>randomConicGradient()  用于生成整个  conic-gradient()  内的参数,也就是每一区间的颜色;</li>     <li>vw  和  vh  是比较新的 CSS 单位,一个页面而言,它的视窗的高度就是 100vh,宽度就是 100vw 。</li>    </ul>    <p>OK,刷新页面,得到如下效果图:</p>    <p><img src="https://simg.open-open.com/show/f2852330de397ac2c5b24b251831bf49.png"></p>    <p>卧槽,很酷炫,bling bling 闪闪发光的感觉啊!而且是随机生成的各种颜色,所以每次刷新都有新体验有木有!!</p>    <p><img src="https://simg.open-open.com/show/2c55d4de850bbd260a6a303de26827f5.gif"></p>    <p>还没完,接下来给它加上旋转动画,蹬蹬蹬,旋转起来大概是这样:</p>    <p><img src="https://simg.open-open.com/show/d833e72b3a0e542e0fc9a329f5b97cb5.gif"></p>    <p>由于 GIF 图大小的限制,只看 GIF 没办法感受到全屏下那种科幻眩晕的感觉,墙裂建议你戳进来看看:</p>    <p><a href="/misc/goto?guid=4959750403554888131" rel="nofollow,noindex">CodePen Demo -- conic-gradient Animation</a></p>    <h2>脑洞时刻</h2>    <p>到这里我还是不是很满足。想到了之前的  mix-blend-mode 属性。</p>    <p>想了解  mix-blend-mode 这个属性,可以戳我看看: <a href="/misc/goto?guid=4959750403664431919" rel="nofollow,noindex">不可思议的颜色混合模式 mix-blend-mode</a></p>    <p>如果多个圆锥渐变层级叠加,并且运用上  mix-blend-mode 会发生什么?好可怕的想法...</p>    <p><img src="https://simg.open-open.com/show/c95d01083de6f2686ab9f83b7cf9a65d.jpg"></p>    <p>最终捣鼓出这种非常科幻的效果:</p>    <p><img src="https://simg.open-open.com/show/207b11316a524e71eed8c2ef77f1ee3d.gif"></p>    <p>上图使用了 2 个半透明的圆锥渐变,相对反向进行旋转,并且在底层使用  mix-blend-mode: overlay 叠加了一个白黑径向渐变图层。可以看看代码及效果:</p>    <p><a href="/misc/goto?guid=4959750403790974024" rel="nofollow,noindex">CodePen Demo -- conic-gradient Animation</a></p>    <h2>在项目中使用  conic-gradient</h2>    <p>上面的例子酷炫归酷炫,但是在项目中实用性不强。那么圆锥渐变是否能用于业务中的?答案是肯定的。</p>    <p>看看下面这个图,芝麻信用分背景渐变颜色条,不使用 JS,纯 CSS 借助  conic-gradient 如何画出来。</p>    <p><img src="https://simg.open-open.com/show/066776487ef3a7b7f4750751b3730dad.jpg"></p>    <p>假设我们的结构如下:</p>    <pre>  <code class="language-css"><div class="bg">      <div class="point"></div>  </div>  </code></pre>    <p>CSS:</p>    <pre>  <code class="language-css">.bg {      position: relative;      margin: 50px auto;      width: 400px;      height: 400px;      border-radius: 50%;      background: conic-gradient(#f1462c 0%, #fc5d2c 12.4%, #fff 12.5%, #fff 12.5%, #fc5d2c 12.5%, #fba73e 24.9%, #fff 24.9%, #fff 25%, #fba73e 25%, #e0fa4e 37.4%, #fff 37.4%, #fff 37.5%, #e0fa4e 37.5%, #12dd7e 49.9%, #fff 49.9%, #fff 50%, #12dd7e 50%, #0a6e3f 62.4%, #fff 62.4%, #fff 62.5%);      transform: rotate(-112.5deg);      transform-origin: 50% 50%;  }    .bg::before {      content: "";      position: absolute;      top: 50%;      left: 50%;      transform: translate(-50%, -50%);      width: 370px;      height: 370px;      border-radius: 50%;      background: #fff;  }    .bg::after {      content: "";      position: absolute;      top: 50%;      left: 50%;      transform: translate(-50%, -50%);      width: 320px;      height: 320px;      border-radius: 50%;      background:           radial-gradient(#fff 0%, #fff 25%, transparent 25%, transparent 100%),          conic-gradient(#f1462c 0 12.5%, #fba73e 0 25%, #e0fa4e 0 37.5%, #12dd7e 0 50%, #0a6e3f 0 62.5%, #fff 0 100%);            }    .point {      position: absolute;      width: 30px;      height: 30px;      transform: translate(-50%, -50%);      left: 50%;      top: 50%;      background: radial-gradient(#467dc6 0%, #a4c6f3 100%);      border-radius: 50%;      z-index: 999;  }    .point::before {      content: "";      position: absolute;      width: 5px;      height: 350px;      left: 50%;      top: 50%;      transform: translate(-50%, -50%) rotate(0);      border-radius: 100% 100% 5% 5%;      background: linear-gradient(          180deg,          #9bc7f6 0,          #467dc6 50%,          transparent 50%,          transparent 100%      );      animation: rotate 3s cubic-bezier(.93, 1.32, .89, 1.15) infinite;  }    @keyframes rotate {   50% {    transform: translate(-50%, -50%) rotate(150deg);   }   100% {    transform: translate(-50%, -50%) rotate(150deg);   }  }    </code></pre>    <p>为了凸显  conic-gradient 的实用性,简单将二者合二为一,模拟了一下。看看效果,大功告成,所以说  conic-gradient 还是有用武之地的:</p>    <p><img src="https://simg.open-open.com/show/94734d2328d088f3f0c06e05e70c6e0d.gif"></p>    <p><a href="/misc/goto?guid=4959750403916279813" rel="nofollow,noindex">CodePen Demo -- 使用 conic-gradient 实现表盘信用分示例</a></p>    <h2>圆锥渐变  conic-gradient polyfill 垫片库</h2>    <p>看到这里,想必读者们都跃跃欲试这么神奇的属性。</p>    <p>但是,按照惯例,这种 “高科技” 通常兼容性都不怎么滴。 conic-gradient 兼容性又如何呢?</p>    <p>非常惨烈,CSS 官方对其的描述是:</p>    <ul>     <li>处于修正阶段的模块(Modules in the revising phase)</li>    </ul>    <p>处于修正阶段的模块没有处于改善阶段的模块稳定。通常它们的语法还需要详细审查,说不定还会有很大的变化,而且不保证和之前的兼容。替代的语法通常经过测试并已经实现。</p>    <p>万幸的是,在文章开头我也提到了,感谢  <a href="/misc/goto?guid=4959750404040441717" rel="nofollow,noindex">LeaVerou</a> 大神,让我们可以提前使用上这么美妙的属性。</p>    <p>LeaVerou 提供了一个垫片库,按照本文上述的语法,添加这个垫片库,就可以开心的使用起来啦。</p>    <p>你需要添加如下的 JS ,垫片库会按照 CSS 语法,生成对应的圆锥渐变图案,并且转化为 BASE64 代码:</p>    <pre>  <code class="language-css"><script src="//cdnjs.cloudflare.com/ajax/libs/prefixfree/1.0.7/prefixfree.min.js"></script>  <script src="//leaverou.github.io/conic-gradient/conic-gradient.js"></script></code></pre>    <p>因为垫片库的作用是将我们的 CSS 语法转化成为 BASE64 代码替换  background-image: url() 中的内容,所以,上线后是不需要再添加这两个库的。</p>    <h2>参考文献</h2>    <p><a href="/misc/goto?guid=4959750404151030849" rel="nofollow,noindex">CSS conic-gradient() polyfill</a></p>    <h2>最后</h2>    <p>系列 CSS 文章汇总在我的  <a href="/misc/goto?guid=4959750404263307362" rel="nofollow,noindex">Github</a> 。</p>    <p>好了,本文到此结束,希望对你有帮助 :)</p>    <p>如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。</p>    <p> </p>    <p>来自:http://www.cnblogs.com/coco1s/p/7079529.html</p>    <p> </p>