创建一个反向剪辑路径 - CSS或SVG

时间:2018-02-11 22:45:01

标签: html css css3 svg clip-path

我试图创建一个实质上与CSS剪辑路径相反的东西。使用剪辑路径时,会剪切图像或div,以便仅保留指定的形状,并有效删除背景的其余部分。

我希望如此,如果我剪辑一个形状,它基本上会在最上层的一个洞中打出并移除形状,而不是背景。这可能吗?我也对SVG解决方案开放,但我是SVG的新手,所以要善待:)

基本上,在下面的代码中,我有一个绝对位于红色正方形内的蓝色方块,并且希望能够从蓝色方块中打出一个形状,以便下面的红色图层显示原来的形状。实际上会有一个图像作为背景图层,因此我无法接受模仿我想要的伪效果,但实际上并没有打出形状。

任何帮助都会很棒!

codepen:https://codepen.io/emilychews/pen/GQmyqx



body {
  width: 100%; 
  height: 100vh;
  padding: 0; margin: 0;
  display: flex;
  }

#box {
  margin: auto;
  position: relative;
  width: 33%;
  height: 200px;
  background: red;
}

#innerbox {
  width: 100%;
  height: 100%;
  background: blue;
  top: 0;
  left: 0;
  position: absolute;
}

<div id="box">
  <div id="innerbox"></div>
</div>
&#13;
&#13;
&#13;

2 个答案:

答案 0 :(得分:3)

您可以将图像置于上方的蓝色部分,然后在其上应用clip-path,那么结果将与您在蓝色部分内创建一个洞以查看相同图片

&#13;
&#13;
body {
  width: 100%; 
  height: 100vh;
  padding: 0; margin: 0;
  display: flex;
  }

#box {
  margin: auto;
  position: relative;
  width: 33%;
  height: 200px;
  background: blue;
}

#innerbox {
  width: 100%;
  height: 100%;
  background: url(https://lorempixel.com/400/400/) center/cover;
  top: 0;
  left: 0;
  position: absolute;
  z-index:1;
  clip-path:polygon(10% 10%, 10% 90%, 90% 50%);
}
&#13;
<div id="box">
  <div id="innerbox"></div>
</div>
&#13;
&#13;
&#13;

另一个想法是考虑多个背景,你会得到比剪辑路径更好的支持,而且代码更少:

&#13;
&#13;
body {
  height: 100vh;
  margin: 0;
  display: flex;
}

#box {
  margin: auto;
  position: relative;
  width: 33%;
  height: 200px;
  background: 
    linear-gradient(to bottom right,transparent 49%,blue 50%) bottom/100% 60%,
    linear-gradient(to top right,transparent 49%,blue 50%) top/100% 60%,
    linear-gradient(blue,blue) left/20% 100%,
    url(https://lorempixel.com/400/400/) center/cover;
  background-repeat:no-repeat;
}
&#13;
<div id="box">
</div>
&#13;
&#13;
&#13;

<强>更新

如果你想要一些不透明度,这里有一个想法,你必须使用clip-path复制内容(一个缺点):

&#13;
&#13;
body {
  width: 100%; 
  height: 100vh;
  padding: 0; margin: 0;
  display: flex;
  }

#box {
  margin: auto;
  position: relative;
  width: 33%;
  height: 200px;
  background: blue;
}

#innerbox,#innerbox-2 {
  width: 100%;
  height: 100%;
  background: url(https://lorempixel.com/400/400/) center/cover;
  top: 0;
  left: 0;
  position: absolute;
  z-index:2;
}
#innerbox {
  /* if you initially planned to have x opacity so you need to set 1-x here*/
  opacity:0.4;
}

#innerbox-2 {
  z-index:1;
  clip-path:polygon(10% 10%, 10% 90%, 90% 50%);
  animation:animate 5s linear alternate infinite;
}

@keyframes animate {
  from {
    clip-path:polygon(10% 10%, 10% 90%, 90% 50%);
  }
  to {
     clip-path:polygon(20% 50%, 90% 50%, 80% 10%);
  }
}
&#13;
<div id="box">
  <div id="innerbox">
    <h1>Title</h1>
    <p>Some content</p>
  </div>
  <div id="innerbox-2">
    <h1>Title</h1>
    <p>Some content</p>
  </div>
</div>
&#13;
&#13;
&#13;

更新2

您可以考虑使用SVG来满足您的初始要求。只需使用SVG而不是div,你将有一个面具。

&#13;
&#13;
body {
  width: 100%; 
  height: 100vh;
  padding: 0; margin: 0;
  display: flex;
  }

#box {
  margin: auto;
  position: relative;
  width: 33%;
  height: 200px;
  background: blue;
  background: url(https://lorempixel.com/400/400/) center/cover;
}

#innerbox {
  width: 100%;
  height: 100%;
  top: 0;
  left: 0;
  position: absolute;
  z-index:1;
}
&#13;
<div id="box">
  <svg viewBox="0 0 200 200" id="innerbox" preserveAspectRatio="none">
  <defs>
    <mask id="hole">
      <rect width="100%" height="100%" fill="white"/>
      <!-- the hole defined a polygon -->
      <polygon points="20,20 20,180 180,100 " fill="black"/>
    </mask>
  </defs>
  <!-- create a rect, fill it with the color and apply the above mask -->
  <rect fill="blue" width="100%" height="100%" mask="url(#hole)" />
</svg>
</div>
&#13;
&#13;
&#13;

您也可以使用相同的SVG作为背景:

&#13;
&#13;
body {
  width: 100%; 
  height: 100vh;
  padding: 0; margin: 0;
  display: flex;
  }

#box {
  margin: auto;
  position: relative;
  width: 33%;
  height: 200px;
  background: blue;
  background: url(https://lorempixel.com/400/400/) center/cover;
}

#innerbox {
  width: 100%;
  height: 100%;
  top: 0;
  left: 0;
  position: absolute;
  z-index:1;
  background:url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 200 200" preserveAspectRatio="none"><defs><mask id="hole"><rect width="100%" height="100%" fill="white"/> <polygon points="20,20 20,180 180,100 " fill="black"/></mask></defs><rect fill="blue" width="100%" height="100%" mask="url(%23hole)" /></svg>');
}
&#13;
<div id="box">
  <div id="innerbox"></div>
  
</div>
&#13;
&#13;
&#13;

答案 1 :(得分:1)

这在Google上排名很高,答案无法解决我的问题,因为无法触摸背景图片,所以这是另一种解决方法:

使用剪切路径创建框架。

body {
  width: 100%;
  height: 100vh;
  padding: 0;
  margin: 0;
  display: grid;
  place-items: center;
}

#clip,
#background {
  width: 400px;
  height: 400px;
}

#clip {
  clip-path: polygon(0% 0%, 0% 100%, 25% 100%, 25% 25%, 75% 25%, 75% 75%, 25% 75%, 25% 100%, 100% 100%, 100% 0%);
  position: absolute;
  background: #fff;
  opacity: 0.8;
}

#background {
  background: url(https://lorempixel.com/400/400/) center/cover;
  z-index: -1;
}
<div id="background">
  <div id="clip"></div>
</div>

为了方便起见,我将clip-div放在图像中,但您也可以将其放在外面。但是,请确保您对limited browser support of clip-path没问题。