如何只使用CSS将应用程序图标图像变为圆角正方形

14

我一直在努力尝试使用The Code Player的CSS3 Squircles示例在我的网站上创建类似于iOS-7风格的应用程序图标(在Safari浏览器中测试)。该示例使用伪类标签来裁剪背景颜色,而我需要围绕<img>进行裁剪。如果您不熟悉,squircle类似于圆角矩形,但是边缘会超出角半径,如下所示:

squircle图片

.icons img {
  width: 100px;
  height: 100px;

  border-radius: 24%;
}

.icons a {
  text-decoration: none;
  display: inline-block;
  position: relative;
}
/*Now we will create fish-eye shapes using pseudo elements and clip them to add a curve to the sides of the icons*/
.icons a:before, .icons a:after {
  content: '';
  position: absolute; left: 0; top: 0;
  width: 100%; height: 100%;
  background: inherit;
  border-radius: 100%; /*circle*/
  /*time to transform the circle into fish-eye shape. iOS7 style now.*/
  -webkit-transform: scaleX(2) scaleY(1.05);
  transform: scaleX(2) scaleY(1.05);
  /*clipping the left and right sides - 17px from the left and right*/
  clip: rect(0, 66px, 100px, 34px);
  /*pushing it behind the icon*/
  z-index: -1;
}
/*duplicating the :before element and rotating it 90deg and swapping the X/Y transforms*/
.icons a:after {
  -webkit-transform: scaleY(2) scaleX(1.05) rotate(90deg);
}
<div class="icons">
  <a href="#"><img src="http://lorempixel.com/256/256/abstract/2/" /></a>
</div>


1
增加边框半径?最终,你会得到一个圆形。 - jmargolisvt
1
@jmargolisvt 我在尝试制作一个“squircle”(不是圆形)。我已更新问题以使其更清晰明了。 - Dov
1
一种方法是使用 clip-path [链接] https://css-tricks.com/clipping-masking-css/ - Mugé
我不知道创建一些圆角div标签并在其中放置图像的问题出在哪里。在iOS上运行得很好。如果你想让角更圆润一些,我同意@jmargolisvt的建议,只需增加border-radius即可。 - Mugé
1
https://pavellaptev.medium.com/squircles-on-the-web-houdini-to-the-rescue-5ef11f646b72 - Bergi
4个回答

16

最简单的解决方案可能是使用透明背景创建图像,直到实现以下某些功能。

如果您可以通过CSS添加图像,则只需将高度、宽度、background-image和background-size添加到链接(.icons a)即可。

注意:这可能不是所需效果,因为它是由背景颜色补充的。

.icons a {
      height: 100px;
      width: 100px;
      background-image: url(https://picsum.photos/256/);
      background-size: cover;
      text-decoration: none;
      color: white;
      display: inline-block;
      margin: 20px;
      border-radius: 24px;
      position: relative;
    }

    .icons a:before,
    .icons a:after {
      content: '';
      position: absolute;
      left: 0;
      top: 0;
      width: 100%;
      height: 100%;
      background: inherit;
      border-radius: 100%;
      -webkit-transform: scaleX(2) scaleY(1.05);
      transform: scaleX(2) scaleY(1.05);
      clip: rect(0, 66px, 100px, 34px);
      z-index: -1;
    }

    .icons a:after {
      -webkit-transform: scaleY(2) scaleX(1.05) rotate(90deg);
      transform: scaleY(2) scaleX(1.05) rotate(90deg);
    }
<div class="icons">
    <a href="#"></a>
</div>

如果不是这种情况,您可以向图像添加大小和边框半径。在这种情况下,伪圆角边框将由“.icon a”元素上的背景颜色填充。

注意: 这可能不是所期望的效果,因为它会与背景颜色相补充。

.icons a {
      height: 100px;
      width: 100px;
      background: red;
      text-decoration: none;
      color: white;
      display: inline-block;
      margin: 20px;
      border-radius: 24px;
      position: relative;
    }
    .icons img{
      height: 100px;
      width: 100px;
      border-radius: 24px;
    }
    .icons a:before, .icons a:after {
      content: '';
      overflow: hidden;
      position: absolute; left: 0; top: 0;
      width: 100%; height: 100%;
      background: inherit;
      border-radius: 100%;
      -webkit-transform: scaleX(2) scaleY(1.05);
      transform: scaleX(2) scaleY(1.05);
      clip: rect(0, 66px, 100px, 34px);
      z-index: -1;
    }
    .icons a:after {
        -webkit-transform: scaleY(2) scaleX(1.05) rotate(90deg);
        transform: scaleY(2) scaleX(1.05) rotate(90deg);
    }
<div class="icons">
    <a href="#">
    <img src="https://picsum.photos/256/">
    </a>
</div>  

SVG解决方案1: 使用SVG进行剪切路径,但是这在WebKit中尚不被支持(将剪切的图像粘在屏幕左上角)。有关更多信息,请参见此链接:https://css-tricks.com/clipping-masking-css/#comment-1587234

#squircle{  
  -webkit-clip-path: url(#svg-shape);
  -moz-clip-path: url(#svg-shape);
  -o-clip-path: url(#svg-shape);
  -ms-clip-path: url(#svg-shape);
  clip-path: url(#svg-shape);
}
<img src="https://picsum.photos/400/" id="squircle">

<svg height="0" width="0" version="1.1"
     xmlns="http://www.w3.org/2000/svg">
  <defs>
  <clipPath id="svg-shape">
<path d="M100,200c43.8,0,68.2,0,84.1-15.9C200,168.2,200,143.8,200,100s0-68.2-15.9-84.1C168.2,0,143.8,0,100,0S31.8,0,15.9,15.9C0,31.8,0,56.2,0,100s0,68.2,15.9,84.1C31.8,200,56.2,200,100,200z" />
    </clipPath>
    </defs>
</svg>

SVG解决方案2:使用图案将图像添加为背景图像。

svg.iOS-svg {
  height: 100px;
  width: 100px;
}
<svg class="iOS-svg" viewBox="0 0 200 200">
  <defs>
    <pattern id="squircle" patternUnits="userSpaceOnUse" width="200" height="200">
        <image xlink:href="https://picsum.photos/256/" x="0" y="0" width="200" height="200" />
    </pattern>
  </defs>

    <path d="M100,200c43.8,0,68.2,0,84.1-15.9C200,168.2,200,143.8,200,100s0-68.2-15.9-84.1C168.2,0,143.8,0,100,0S31.8,0,15.9,15.9C0,31.8,0,56.2,0,100s0,68.2,15.9,84.1C31.8,200,56.2,200,100,200z" fill="url(#squircle)" />
</svg>

其他资源: http://caniuse.com/#search=clip-path (撰写时部分支持) SVG支持: http://caniuse.com/#search=svg


不要用 -webkit 替换 transform, 而是在普通的 transform 上方加入该规则,这样当浏览器不基于 webkit 时,它就可以作为回退。您可能还想为 IE9 添加 -ms 规则和为 Firefox 添加 -moz 规则。 您可能还需要添加前缀到其他规则,例如:border-radiusbackground-size,虽然现在许多浏览器都支持它们。 如果您使用 Sass,则此链接可能会让您更好地了解跨浏览器选择器以及如何使用 autoprefixer: 链接,此链接可以查看哪些浏览器支持 CSS3 规则:链接 - Diego
是的,我提到了这一点,因为我不确定真正的用途是什么。你是否依赖于img标签的存在? - Diego
我已经更新了你的答案,加入了webkit扩展和一个新的多彩示例图像。我不是“依赖”于img标签,但我更喜欢在HTML中尽可能地指定图像资源。如果没有简单的方法来实现它,我肯定会使用你的第一个示例,因为它确实有效。 - Dov
已更新答案,添加了一些额外选项,尽管它们可能尚未完全受到所有浏览器的支持。 - Diego
让我们在聊天中继续这个讨论:http://chat.stackoverflow.com/rooms/91644/discussion-between-dov-and-diego。 - Dov
显示剩余3条评论

0

.squircle {
  width: 52.1em;
  height: 52.1em;
  position: relative;
  display: inline-block;
  margin: 1em auto;
  vertical-align: middle;
  -webkit-transform: scale(0.5);
  -webkit-transform-origin: 50% 0;
}
.squircle > * {
  position: absolute;
  width: 100%;
  display: inline-block;
  height: 100%;
  background: 50% 50% no-repeat;
  z-index: 5;
}
.squircle:before,
.squircle:after,
.squircle > *:before,
.squircle > *:after {
  position: absolute;
  background: #00aeef;
}
.squircle:before {
  top: 0;
  left: 4em;
  border-top-left-radius: 20em 10em;
  width: 50%;
  bottom: 0;
  border-bottom-left-radius: 20em 10em;
  content: "";
}
.squircle:before {
  top: 0;
  left: 4em;
  border-top-left-radius: 80% 10em;
  width: 50%;
  bottom: 0;
  border-bottom-left-radius: 80% 10em;
  content: "";
}
.squircle:after {
  top: 0;
  bottom: 0;
  right: 4em;
  border-top-right-radius: 80% 20%;
  border-bottom-right-radius: 80% 20%;
  width: 25em;
  content: "";
}
.squircle > *:before {
  top: 4em;
  bottom: 4em;
  border-top-left-radius: 100% 50%;
  border-bottom-left-radius: 100% 50%;
  width: 10em;
  content: "";
}
.squircle > *:after {
  top: 4em;
  bottom: 4em;
  right: 0;
  border-top-right-radius: 100% 50%;
  border-bottom-right-radius: 100% 50%;
  width: 10em;
  content: "";
  z-index: 4;
}
body {
  background: #1F1A1D;
}
body::before {
  height: 100%;
  content: '';
  width: 0;
  background: red;
  vertical-align: middle;
  display: inline-block;
}
<span class="squircle">
  <span>I'm a squircle!</span>
</span>


0

-1

<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="470px" height="468px" viewBox="0 0 470 468" version="1.1">
  <path fill="#987EFC" fill-rule="evenodd" d="M9.635 132.808C24.782 59.782 71.388 19.109 144.085 6.822c53.74-9.081 107.5-9.196 161.15.255 74.852 13.185 119.85 56.23 134.185 130.36 11.075 57.29 11.249 115.191-.174 172.427-15.324 72.52-63.132 117.285-135.561 129.527-53.74 9.08-107.5 9.195-161.15-.255-74.852-13.186-120.05-58.38-134.384-132.509-11.64-57.668-10.52-115.935 1.484-173.82z" id="path-1"/>
  </svg>


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接