每日分享前端插件干货,欢迎关注!
现代CSS技术是美化网页设计过程的最佳方式之一。如果你一直在使用CSS,那么肯定遇到过关于布局或跨浏览器兼容性方面的问题。例如,CSS3样式不适用于旧版本的Internet Explorer。
同时,随着网络技术的发展,开发人员也在努力应对其他问题。例如,利用其他库来优化大页面网站的加载时间,或使div
标签更具响应性,而无需过多依赖Bootstrap。随着时代的进步和挑战的增加,CSS也在不断发展。
本次高级CSS教程将讨论12个现代CSS技术用于克服CSS带来的旧问题。
如果你打算开发可用于其他页面或UI组件的可重用CSS文件,那么了解CSS选择器的工作原理至关重要。
现在让我们来看看一些高级选择器。
通用选择器可用于网站的所有元素。例如,如果你想为页面中的所有元素设置特定的边距,那么需要这样写:
* {
margin: 1.5rem;
}
如果想将特定样式应用于某些具有相似属性但类名不同的元素,怎么办?可以使用属性选择器。
[class*="component_"] {
border: 2px solid blue;
}
属性选择器将对类名为component_title
和component_label
的元素应用边框。
如果你想将导航栏中菜单的字体大小设置为粗体但导航内容是动态生成的,怎么办?它们没有任何特定的类。但你可以使用子组合选择器。
.navigation-menu > ul > li > a {
font-weight: bold;
}
不同颜色交替的表格怎么实现?猜猜如何只用几行CSS就做到这一点吗?答案是伪类。
tbody tr:nth-child(odd) {
background-color: green;
}
代码将对表格中的奇数行应用绿色背景。
除了odd
或even
,还可以使用n+1
或3n+1
等属性应用不同的属性到不同的行。
bootstrap网格解决了在不同分辨率的设备中均匀分布响应容器的问题。但随着时间的推移,只用于网格的库越发繁重,因此开发人员需要另一种解决方案。
现代CSS技术,例如网格,为我们提供了一种简化的解决方案。
$minColumnWidth: 10rem;
.grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax($minColWidth, 1fr));
grid-gap: 2rem;
& + .grid {
margin-top: $gridGap;
}
}
这些代码做了什么?
首先定义了网格元素的最小宽度。
grid-template-columns
中使用了repeat
函数,以便CSS将相同的参数应用于每一列。此外,使用auto-fit
而不是数字的原因是,列将具有相同的宽度——无论使用多少列,都会拉伸并填充可用空间。
代码中还使用了minmax()
函数来设置每列的最大宽度,并将1fr
作为最大值。这可以确保内容填满一列直至达到总可用空间。
最后还添加了gap
和可选规则,以在连续容器之间应用相同的值。
结果如下:
只有一个缺点。对于三个以上的列,在某些视图中,可能会出现孤立的列。
我们可以通过媒体查询来解决问题,但需要为多个媒体尺寸编写大量代码。
曾几何时,我们通过jQuery计算以确保容器高度相等。那时,两个容器的内容不相等的时候,我们就想办法让它们的高度等于UX。
但是现代CSS技术——网格系统——可以让这个过程变得更简单。
/*Parent container*/
.parent {
display: grid;
grid-auto-flow: column;
}
/*Child elements*/
.child {
height: 100%;
}
通过将grid-auto-flow
切换为row
或column
,我们可以拥有统一宽度或高度的容器。
CSS flexbox允许我们通过更简单的代码来使用相同的功能。
/*Parent elements*/
.parent {
display: flex;
/* Child elements */
.child {
height: 100%;
}
}
那么,哪一个更好呢?答案是,两者各有优势。如果你想解决等高元素的问题,那么flexbox更优。但是,在flexbox中,元素的宽度可能并不相等。
当然,如果你需要,网格也可以处理等宽的元素。此外,如果想要每一行拥有最大列数,则可以使用网格处理数学。但是如果你使用flexbox,则需要为此定义计算。
虽然现如今back to top
链接用的少了,但曾经有一段时间确实很流行。方法是创建一个链接,允许用户在网站顶部向后滚动。那么如何使用现代CSS技术开发back-to-top
链接呢?
先创建内容。
Title
添加smooth滚动到页面。
html {
scroll-behavior: smooth;
}
然后根据喜好设置back-to-top
链接的样式。你也可以添加漂亮的SVG图标替换手动样式。
下面选择滚动条的行为方式。定义滚动长度:
$scrollLength: 100vh;
在包装器中将滚动值设置为预定义的滚动长度:
.back-to-top-wrapper {
position: absolute;
top: $scrollLength;
}
给back-to-top
链接添加必要的样式。记住要使用position-sticky
以将其保持在特定位置。
.back-to-top-link {
// using 'fixed' as fallback when `sticky` not supported
position: fixed;
position: sticky;
// achieves desired positioning within the viewport
// relative to the top of the viewport once `sticky` takes over, or always if `fixed` fallback is used
top: calc(100vh - 5rem);
// ... other styles
}
完成!
可访问性是网站的必备功能,且必须遵循WCAG指南。
通常,在按钮或链接上移除:focus
可以摆脱原生样式。然而,在这样做的同时,因为未能提供替代的焦点状态,结果就是使用键盘导航站点的人无法检测到当前的焦点元素。直接违背了可访问性的原则。
我们可能会觉得浏览器的默认:focus
样式,使那些保存于页面的漂亮元素显得不那么好看。但是,它是为备用焦点样式添加代码的理想选择。例如,假设有一个按钮:
你可以添加:
button:focus {
outline: max(1px, 0.1em) solid currentColor;
outline-offset: 0.25em;
}
将currentColor
视为页面的主题颜色。在本例中为绿色。
使用outline-offset
,可以调整轮廓与元素的间距:
排版是增强网页外观和视觉效果的最佳方式。
不知道应该使用%
、rem
、em
还是px
?在定义排版时你需要做的第一件事是——忘记px
。因为当用户在浏览器中放大或缩小时,px
不会按比例缩放。推荐单位是rem
。
rem
的默认值为16 px
,当用户在浏览器中放大时,用rem定义的字体大小保持一致。
em
与元素的字体大小规则成比例。因此只能在你希望子元素相对于父元素的间距行为时使用。
%
的行为与em
几乎相似。但是,当你需要相对大小时,em
更可取。
防止文本溢出是使网站面向未来的好方法。它确保了如果将来容器中的某些文本增加,文本也不会超出容器或其边界。
p,
li,
h1,
h2,
h3,
h4 {
// Help prevent overflow of long words/names/URLs
word-break: break-word;
}
如果你在全局范围内选择了较大的字体大小,那么可能会在小型设备上遇到溢出问题。我们可以通过流体类型解决这个问题。即定义一个根据视口调整的字体大小值,就像响应式图像一样。可以使用代码来计算最小和基本字体大小。
font-size: unquote("min(max(#{$fluid-min}rem, #{$fluid-scaler}), #{$level-size})");
或者可以使用vw
(视口宽度)作为字体单位。
开发人员需要自定义样式的单选按钮,因为默认的HTML单选按钮在不同浏览器中的显示方式不同。以下是在Mac上运行的Firefox中单选按钮的外观。
这是它们在Safari中的外观。
不仅如此,原生单选按钮不能随字体大小缩放。
解决方案是创建统一的单选按钮,该按钮在所有浏览器上的显示都相同,并且与标签的字体大小一致。
先从HTML开始:
接着是CSS。我们使用SCSS来为单选按钮设置主题。
创建自定义颜色变量。
:root {
--color: green;
}
接下来使用通用选择器来重置box-sizing
方法。它将在计算任意元素的最终大小时包括边框和填充。
*,
*:before,
*:after {
box-sizing: border-box;
}
我们在标签中添加了.radio
类。使用grid-gap
将在按钮和标签之间添加一些间距。我们还将使用:focus-within
和transform
以便当该特定选项处于焦点时,标签的尺寸会扩大。
.radio {
display: grid;
grid-template-columns: min-content auto;
grid-gap: 0.5em;
font-size: 2.25rem;
color: var(--color);
&:focus-within {
.radio-label {
transform: scale(1.05);
opacity: 1;
}
}
}
调整行高,添加过渡,并在选项未处于焦点时,降低标签的不透明度。
.radio-label {
line-height: 1;
transition: 180ms all ease-in-out;
opacity: 0.8;
}
对包装自定义控件和本机输入的按钮添加display-flex
。
对焦点状态使用双框阴影,并确保基本按钮和焦点状态之间存在差异。
.radio-input {
display: flex;
input {
opacity: 0;
width: 0;
height: 0;
&:focus + .radio-control {
box-shadow: 0 0 0 0.05em #fff, 0 0 0.15em 0.1em currentColor;
}
}
}
添加按钮处于选中阶段时的样式。
.radio-gradient input:checked + .radio-control {
background: radial-gradient(currentcolor 50%, rgba(255, 0, 0, 0) 51%);
}
创建:before
元素,以设置按钮未选中时的动画。
.radio-before {
.radio-control {
display: grid;
place-items: center;
}
input + .radio-control::before {
content: "";
width: 0.5em;
height: 0.5em;
box-shadow: inset 0.5em 0.5em currentColor;
border-radius: 50%;
transition: 180ms transform ease-in-out;
transform: scale(0);
}
input:checked + .radio-control::before {
transform: scale(1);
}
}
最后,添加样式到单选按钮和正文。
.radio-control {
display: block;
width: 1em;
height: 1em;
border-radius: 50%;
border: 0.1em solid currentColor;
transform: translateY(-0.05em);
}
body {
min-height: 100vh;
display: grid;
place-content: center;
grid-gap: 2rem;
padding: 1rem;
}
最终结果如下:
虽然单页应用程序不会有太多这方面的问题,但如果你正在创建静态网页,那么你可能会发现页脚经常浮动。
我们可以通过两种方法摆脱这个问题。
代码如下:
body {
min-height: 100vh;
display: grid;
grid-template-rows: auto 1fr auto;
}
代码先保持body
的最小高度。然后使用grid-template-rows
正确分隔内容。该方法使用小数单元来计算可用空间并将其分配到多行。因此,它将填充页眉和页脚之间的所有可用空间。
使用flexbox,方法更简单。
body {
min-height: 100vh;
display: flex;
flex-direction: column;
}
首先,通过min-height
确保body
伸展到屏幕的全高。然后,设置flex-direction
保留堆叠的块元素并保持文档行为正常。
以前用户需要很艰难地使用jQuery和position:absolute
为图像标题和图像大小设置动画。
但是,现如今,只需几行代码就可以为标题设置动画并在鼠标悬停图片时触发动画。请看:
.image-caption {
// ... existing styles
transition: transform 500ms ease-in;
will-change: transform;
}
The transition property helps to animate the caption. We can trigger the animation on hover using -
.image:hover {
.image-caption {
transform: translateY(0);
}
}
解决了鼠标悬停的问题。那么,如果有人使用键盘浏览网页怎么办。在这种情况下,我们可以使用focus
取代hover
。
.image:focus {
outline: none;
}
.image:hover,
.image:focus {
.image-caption {
transform: translateY(0);
}
感谢CSS3的border-radius
属性,现在的开发人员可以很从容地创建圆角框。例如:
border-radius: 20% 50%;
这形状有点奇怪,但这里的目的是演示边界半径的使用;如果应用计算值,则可以为容器提供所需的任何形状。例如:
border-radius: 3vw 4vw 8vw 2vw;
我们可以使用相对于元素大小的百分比值。如果要保持一致性,像vw
(视口单位)这样的相对单位会很有帮助。这样一来,我们就有了在移动设备上会变小,在桌面上会变大,但保持一致的圆形。
还有一个有趣的属性可以帮助我们为页面元素添加良好和舒缓的用户体验,那就是box-shadow
。通常盒状阴影用于添加卡片或按钮等元素的立体化。例如:
box-shadow:3px 4px 5px 0px rgba(0, 0, 0, 0.38);
这里水平偏移了3px,垂直偏移了4px,还有5px的模糊半径、无扩散半径和灰色阴影。你可以根据自己的选择调整值以增强框的外观。
box-shadow
的一个独特功能是,可以添加多个图层。例如:
box-shadow: 2px 4px 0 4px yellowgreen, 4px 8px 0 8px yellow, 8px 10px 0 12px red;
你知道如何使用box-shadow
为图像添加晕影效果吗?答案是,使用嵌入阴影。
.image-container {
width: 30vmin;
height: 30vmin;
box-shadow: inset 0 0 4vmin 3vmin rgba(0, 0, 0, 0.5);
}
早些时候,如果我们需要将元素保持在页面的中心,必须使用绝对定位并定义left
、right
、top
和bottom
值。好吧,这样痛苦的日子已经一去不复返了。现代CSS技术为我们提供了一种简单的解决方案,可以帮助在多种分辨率下保持居中位置。
使元素水平和垂直居中,只需向元素添加两个属性。
display: grid;
place-content: center;
如果你想使用flexbox
以便与页面上其他flexbox元素保持一致性,则需要三行代码。
display: flex;
align-items: center;
justify-content: center;
如果想将元素放置在任何特定轴的中心对齐,该怎么办?在这种情况下,必须删除display
属性。对于垂直居中对齐:
align-content:center;
结果如下:
而如果想要保持元素水平居中,而不是对齐内容,须使用justify-content
:
justify-content:center;
上面提到的解决方案是针对gird布局的。对于flex布局,代码有所不同。
垂直对齐的代码是这样的:
justify-content: center;
flex-direction: column;
For horizontal alignment, you will need -
justify-content: center;
现代CSS技术帮助我们摆脱了使用脚本来绘制对象的麻烦。现在有很多方法可用于创建基本的CSS形状。下面让我们看看如何创建CSS三角形。
border
属性可以帮助创建三角形。先从容器开始:
.triangle {
border-style: solid;
border-color: transparent;
}
使用border-width
属性并为边框的左侧赋予颜色。
.triangle {
border-style: solid;
border-color: transparent;
/* top | right | bottom | left */
border-width: 7px 0 7px 10px;
border-left-color: blue;
}
结果如下所示:
我们可以打乱border-width
的top
、right
、bottom
和left
值来改变三角形的方向。
线性渐变可以与background-image
属性一起使用。
首先创建容器,给出尺寸并阻止background-repeat
。
.triangle {
width: 8em;
height: 10em;
background-repeat: no-repeat;
}
接着添加渐变。使容器的一半呈蓝色。
添加两个渐变后,形状像容器中的镜像三角形。
background-image: linear-gradient(32deg, blue 50%, rgba(255, 255, 255, 0) 50%), linear-gradient(148deg, blue 50%, rgba(255, 255, 255, 0) 50%);
现在,让我们试着让其看起来像一个三角形。更改背景大小:三角形将有100%的宽度和50%的高度。
background-size: 100% 50%;
最后,为了防止重叠,必须添加背景位置,因为两个渐变的默认位置都是0 0
。
background-position: top left, bottom left;
现在的形状看起来像三角形了。但是,这种方法有一个缺点。图形不会响应。如果纵横比发生变化,则可能需要重新计算度数。
CSS正在快速发展。开发人员正致力于创建纯CSS库,以减少对脚本的依赖并缩减网站的大小,使其对SEO更加友好。
希望本次高级CSS教程可以帮助大家更好地了解现代CSS技术。感谢大家的阅读!
(文本完)
每日分享前端插件干货,欢迎关注!