CSS“inverted border-radius”在元素边界框外创建手机缺口设计[重复]

Posted

技术标签:

【中文标题】CSS“inverted border-radius”在元素边界框外创建手机缺口设计[重复]【英文标题】:CSS "inverse border-radius" outside element's bounding box to create a mobile phone notch design [duplicate] 【发布时间】:2020-04-04 05:51:19 【问题描述】:

我正在尝试使用 html 和 CSS 创建一个看起来像手机的东西,我希望相机有一个类似于“倒置边框半径”的东西,可以将它与框架平滑地连接起来。

我不能只是让它变大并用带有白色背景的伪元素来掩盖不需要的区域,因为屏幕内容可能并不总是白色的。

另外,我不能在同一个元素上使用mask-image,因为“倒置边框半径”实际上会延伸到它的边界框之外,所以我实际上会添加更多的区域而不是减去(加上支持真的很低)。

如果可能,我想避免使用 SVG。

body 
  position: relative;
  overflow: hidden;
  height: 100vh;
  margin: 0;


.phone 
  width: 420px;
  height: 800px;
  padding: 12px 12px 24px;
  position: absolute;
  top: 32px;
  left: 50%;
  transform: translate(-50%, 0);
  background: #000;
  box-shadow: 0 8px 32px 0 rgba(0, 0, 0, .125);
  border-radius: 16px;


.screen 
  height: 100%;
  overflow: hidden;
  position: relative;
  background: #FFF;
  border-radius: 8px;


.viewport 
  height: 100%;
  position: relative;
  overflow-x: hidden;
  overflow-y: scroll;


.notch 
  top: 12px;
  left: 50%;
  width: 24px;
  height: 12px;
  z-index: 10;
  position: absolute;
  transform: translate(-50%, 0);
  background: #000;
  border-bottom-left-radius: 1024px;
  border-bottom-right-radius: 1024px;


.camera 
  top: 0;
  left: 50%;
  width: 12px;
  border: 4px solid #33244A;
  height: 12px;
  position: absolute;
  transform: translate(-50%, -50%);
  background:  #304A58;
  border-radius: 1024px;
  box-sizing: border-box;
<div class="phone">
  <div class="notch">
    <div class="camera"></div>
  </div>

  <div class="screen">
    <div class="viewport"></div>
  </div>
</div>

【问题讨论】:

相关(可能重复)***.com/q/50402503/8620333 @TemaniAfif 不是真正的重复,但相似。在另一个问题中,OP 想要创建的形状由椭圆组成,这使得我在此处添加的一些更简单的解决方案无效。对于像这种更简单的形状,可以使用更多的替代方法。 您同意重复项,方法是在另一个问题中添加您的答案,详细说明完全相同的方法,并显示最后它是同一个问题,唯一的区别是形状略有不同。圆是省略号的一种特殊情况。 【参考方案1】:

有四种方法可以做到这一点,从简单到复杂:

radial-gradient添加2个伪元素。

最简单且支持良好的解决方案。可能是我会使用的那个。

使用mask-image 添加2个伪元素(与上面相同,但支持更差)。

在代码方面与预览版非常相似,但带有really bad support(需要浏览器前缀才能支持它)。

使用border-radiusbox-shadowbackground: transparent 添加2 个伪元素。

需要更多代码,但看起来更流畅,至少在Chrome Version 78.0.3904.108 上,所以也许对你来说是值得的,尽管差异很小。在任何情况下,您可以做的形状都不能像以前的替代品那样复杂,特别是如果您想使用椭圆而不是圆形,就像在另一个问题中一样:https://***.com/a/59278227/3723993。

使用 SVG。

我认为 SVG 解决方案在这里不值得,但对于更复杂的形状或动画/过渡形状来说,它会是一个不错的选择。

您可以在这里查看前 3 个解决方案:

const notch = document.getElementById('notch');
const button = document.getElementById('button');
const xrayCheckbox = document.getElementById('xrayCheckbox');
const xrayLabel = document.getElementById('xrayLabel');
const label = document.getElementById('label');

const solutions = [
  name: 'pseudoelements + radial-gradient',
  classes: 'notch notch-gradient'
, 
  name: 'pseudoelements + box-shadow',
  classes: 'notch notch-shadow'
, 
  name: 'pseudoelements + mask-image',
  classes: 'notch notch-mask'
];

let currentSolutionIndex = 0;
let currentSolution = solutions[currentSolutionIndex];
let xRayEnabled = false;

button.onclick = () => 
  currentSolutionIndex = (currentSolutionIndex + 1) % solutions.length;
  currentSolution = solutions[currentSolutionIndex];
  
  updateLabels();
;

xrayCheckbox.onchange = () =>   
  xRayEnabled = xrayCheckbox.checked;
  
  updateLabels();
;

function updateLabels() 
  if (xRayEnabled) 
    notch.className = `$ currentSolution.classes -xray`;
    label.innerText = `$ currentSolution.name  (X-Ray)`;
    xrayLabel.innerText = 'Disable X-Ray';
   else 
    notch.className = currentSolution.classes;
    label.innerText = currentSolution.name;
    xrayLabel.innerText = 'Enable X-Ray';
  
body 
  position: relative;
  overflow: hidden;
  height: 100vh;
  margin: 0;


.phone 
  width: 420px;
  height: 800px;
  padding: 12px 12px 24px;
  position: absolute;
  top: 32px;
  left: 50%;
  transform: translate(-50%, 0);
  background: #000;
  box-shadow: 0 8px 32px 0 rgba(0, 0, 0, .5);
  border-radius: 16px;


.screen 
  height: 100%;
  overflow: hidden;
  position: relative;
  background: #FFF;
  border-radius: 8px;


.viewport 
  height: 100%;
  position: relative;
  overflow-x: hidden;
  overflow-y: scroll;


.notch 
  top: 12px;
  left: 50%;
  width: 24px;
  height: 12px;
  z-index: 10;
  position: absolute;
  transform: translate(-50%, 0);
  background: #000;
  border-bottom-left-radius: 1024px;
  border-bottom-right-radius: 1024px;


.notch::before,
.notch::after 
  top: 0;
  width: 8px;
  height: 8px;
  content: "";
  position: absolute;


.notch-gradient-xray,
.notch-shadow-xray,
.notch-mask-xray 
  background: red;


/* RADIAL GRADIENT SOLUTION */

.notch-gradient::before 
  left: -6px;
  background: radial-gradient(circle at bottom left, transparent 0, transparent 70%, black 70%, black 100%);


.notch-gradient::after 
  right: -6px;
  background: radial-gradient(circle at bottom right, transparent 0, transparent 70%, black 70%, black 100%);


.notch-gradient-xray::before 
  left: -6px;
  background: green radial-gradient(circle at bottom left, transparent 0, transparent 70%, cyan 70%, cyan 100%);


.notch-gradient-xray::after 
  right: -6px;
  background: green radial-gradient(circle at bottom right, transparent 0, transparent 70%, cyan 70%, cyan 100%);


/* BOX-SHADOW SOLUTION */

.notch-shadow::before 
  left: -6px;
  background: transparent;
  border-radius: 0 8px 0 0;
  box-shadow: 0 -4px 0 0 #000;


.notch-shadow::after 
  right: -6px;
  background: transparent;
  border-radius: 8px 0 0 0;
  box-shadow: 0 -4px 0 0 #000;


.notch-shadow-xray::before 
  left: -6px;
  background: green;
  border-radius: 0 8px 0 0;
  box-shadow: 0 -4px 0 0 cyan;


.notch-shadow-xray::after 
  right: -6px;
  background: green;
  border-radius: 8px 0 0 0;
  box-shadow: 0 -4px 0 0 cyan;


/* MASK SOLUTION */

.notch-mask::before 
  left: -6px;
  background: #000;
  -webkit-mask-image: radial-gradient(circle at bottom left, transparent 0, transparent 70%, black 70%, black 100%);


.notch-mask::after 
  right: -6px;
  background: #000;
  -webkit-mask-image: radial-gradient(circle at bottom right, transparent 0, transparent 70%, black 70%, black 100%);


.notch-mask-xray::before 
  left: -6px;
  background: cyan;
  -webkit-mask-image: radial-gradient(circle at bottom left, transparent 0, transparent 70%, black 70%, black 100%);


.notch-mask-xray::after 
  right: -6px;
  background: cyan;
  -webkit-mask-image: radial-gradient(circle at bottom right, transparent 0, transparent 70%, black 70%, black 100%);


.camera 
  top: 0;
  left: 50%;
  width: 12px;
  border: 4px solid #33244A;
  height: 12px;
  position: absolute;
  transform: translate(-50%, -50%);
  background:  #304A58;
  border-radius: 1024px;
  box-sizing: border-box;


#button 
  font-family: monospace;
  font-size: 16px;
  padding: 8px 16px;
  margin: 32px auto 16px;
  background: transparent;
  border: 2px solid black;
  display: block;
  border-radius: 2px;


#xray 
  font-family: monospace;
  font-size: 16px;
  padding: 0 16px;
  text-align: center;
  display: block;
  margin: 0 0 16px;
  display: flex;
  align-items: center;
  justify-content: center;


#xrayCheckbox 
  margin: 0 8px 0 0;


#label 
  font-family: monospace;
  font-size: 16px;
  padding: 0 16px;
  text-align: center;
<div class="phone">
  <div id="notch" class="notch notch-gradient">
    <div class="camera"></div>
  </div>

  <div class="screen">
    <div class="viewport">
      <button id="button">Change Solution</button>
      
      <label id="xray">
        <input id="xrayCheckbox" type="checkbox" />
        <span id="xrayLabel">Enable X-Ray</span>
      </label>
      
      <div id="label">pseudoelements + radial-gradient</div>
    </div>
  </div>
</div>

【讨论】:

以上是关于CSS“inverted border-radius”在元素边界框外创建手机缺口设计[重复]的主要内容,如果未能解决你的问题,请参考以下文章

css [css:fadeout / fadein] css示例。 #css

css基础 CSS 组合选择符CSS 伪类CSS 伪元素

css 深度提示#css中的css base builder CSS

css基础 CSS 媒体类型CSS 属性 选择器CSS 表单CSS 计数器

什么是css

测开之CSS・第一篇《CSS语法基础》