乐趣区

关于前端:你不应该依赖CSS-100vh这就是原因

微信搜寻【大迁世界】, 我会第一工夫和你分享前端行业趋势,学习路径等等。
本文 GitHub https://github.com/qq449245884/xiaozhi 已收录,有一线大厂面试残缺考点、材料以及我的系列文章。

如果有一个文本和一个按钮,咱们想让文本粘在下面,而按钮粘在上面!应用 CSS Flex 仿佛很容易做到。

// HTML
<div className="layout">
  <p>Lorem ipsum dolor sit amet...</p>
  <button>Sign Up</button>
</div>

// CSS
.layout {
  display: flex;
  flex-direction: column;
  justify-content: space-between;
  min-height: 100vh;
}

在真机检查一下成果:

酷! Git add, git commit, git push, oh yeah!

这有什么问题吗?

当然,是有的! 要看到这个问题,你须要在实在的手机或模拟器上查看你的应用程序。在本文中应用的 iPhone 13(iOS 15.2)进行测试,上面是后果:

啥,底部按钮跑哪里去了?

顺便说一下,它在安卓手机上甚至不能按预期工作。

为什么 100vh 问题会产生在挪动设施上?

我对这个问题进行了一番考察,发现了其中的起因。简短的答案是,浏览器的工具栏高度没有被思考在内。如果你想深刻理解为什么会产生这种状况,Stack Overflow 的这个帖子很有帮忙。

如何修复挪动设施上的 100vh 问题?

第一个倡议是尽量少用 vh。例如,在下面的代码中,你能够应用一个 sticky 按钮,防止应用 vh 单位。

// HTML
<div className="layout">
  <p>Lorem ipsum dolor sit amet...</p>
  <button>Sign Up</button>
</div>

// CSS
.layout {
  display: flex;
  flex-direction: column;
  justify-content: space-between;
  min-height: 100vh;
}
.layout button {
  position: sticky;
  bottom: 0;
}

成果:

它在横向模式下也很好:

说实话,后果是好的,但你不能总是用 sticky 元素来解决 100vh 的问题。

仅应用 CSS 在挪动设施上修复 100VH 问题

时,应用 vh 的目标是为了简略地创立与视口高度相等的局部。例如,当你在建设登陆页面时,这很常见。在这些状况下,position sticky不会有帮忙,这里介绍一下 fill-available属性。它用起来很简略,只有记住应用前缀和回退值就能够了。

.layout {
  min-height: 100vh;            /* fall-back */
  min-height: -moz-available;
  min-height: -webkit-fill-available;
  min-height: fill-available;
}

成果:

而且,当你旋转设施时,它还会更新高度,太棒了!

fill-available 修复 100vh 的问题的确很间接,但在考察这个解决方案时,也遇到过一些问题。

1. HTML 类型申明问题

页面上有 <!DOCTYPE html> 申明,会使 fill-available 在 Chrome 浏览器上无奈失常工作。

甚至不能在安卓浏览器上工作:

因而,为了解决这个问题,必须从页面中删除 doctype 申明。

2. Safari 上的垂直 padding 问题

min-height(或 height)为 fill-available的元素上增加垂直 padding(bottom 和 top),Safari 浏览器上会导致问题,高度不会正确。

要解决这个问题,只需将你的内容包在另一个 div 元素内,就能够了:

// HTML
<div class="screen">
  <div class="content">
    ...
  </div>
</div>

// CSS
.screen {
  background-color: mediumpurple;
  min-height: 100vh;
  min-height: -moz-available;
  min-height: -webkit-fill-available;
  min-height: fill-available;
}
.content {
  color: #fff;
  display: flex;
  flex-direction: column;
  justify-content: center;
  height: 100%;
  padding: 30px;
}

3. fill-available 不能与 calc() 一起应用

须要留神的一件事是,不能在 fill-available 属性下应用 calc()。所以,上面的 CSS 规定就不会失效:

min-height: calc(-webkit-fill-available / 2);

例如,如果须要在元素上有一半的可用高度,必须应用 JavaScript。

应用 JavaScript 修复挪动设施上的 100vh 问题

能够应用 window 的 innerHeight 属性,将元素 height(或minHeight)设置为window.innerHeight,如下所示:

<!DOCTYPE html>
<html lang="en">
<head>
  <style>
    ...
  </style>
</head>
<body>
<div id="intro">
  <h1>Hello World!</h1>
  <h2>The height of this area is equal to...</h2>
</div>
...
<script>
  (function () {const el = document.getElementById('intro');
    el.style.minHeight = window.innerHeight + 'px';
  })();
</script>
</body>
</html>

成果:

接着,再介绍一种花销的形式。一些开发者喜爱依据窗口的外部高度定义一个CSS 变量,并应用该变量来设计他们所需的元素。代码如下:

// 以像素为单位计算 1vh 值
// 基于窗口的外部高度
var vh = window.innerHeight * 0.01;

//  将 CSS 变量设置为根元素
// 相当于 1vh
document.documentElement.style.setProperty('--vh', vh + 'px');

在 CSS 中:

min-height: calc(var(--vh) * 100);

最初一件事是当窗口被调整大小或设施方向扭转时,从新计算这个值:

function calculateVh() {
  var vh = window.innerHeight * 0.01;
  document.documentElement.style.setProperty('--vh', vh + 'px');
}

// 初始计算
calculateVh();

// 调整大小时从新计算
window.addEventListener('resize', calculateVh);

// 在设施方向扭转时从新计算
window.addEventListener('orientationchange', calculateVh);

在我看来,你应该先用 CSS 的解决方案。

代码部署后可能存在的 BUG 没法实时晓得,预先为了解决这些 BUG,花了大量的工夫进行 log 调试,这边顺便给大家举荐一个好用的 BUG 监控工具 Fundebug。

作者:Mehdi Namvar 译者:前端小智 起源:mediun

原文:https://ilxanlar.mdium.com/yo…

交换

有幻想,有干货,微信搜寻 【大迁世界】 关注这个在凌晨还在刷碗的刷碗智。

本文 GitHub https://github.com/qq449245884/xiaozhi 已收录,有一线大厂面试残缺考点、材料以及我的系列文章。

退出移动版