Heim >Web-Frontend >CSS-Tutorial >Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

藏色散人
藏色散人nach vorne
2022-11-25 16:43:122539Durchsuche

In diesem Artikel erfahren Sie, wie Sie das durch die Verwendung von Verlaufsgrafiken verursachte Aliasing-Problem lösen können. Sobald Sie es wissen, können Sie es verwenden. Schauen wir uns an, wie Sie es erreichen können sei für alle nützlich!

Technik zum Verschwinden gezackter CSS-Verläufe

In CSS kann man sagen, dass der Farbverlauf (Gradient) eine der mächtigsten Eigenschaften ist.

Allerdings stoßen Schüler bei der Verwendung von Farbverläufen häufig auf das Aliasing-Problem, das durch Verlaufsgrafiken verursacht wird. [Empfohlenes Lernen: CSS-Video-Tutorial]

Was ist Gradienten-Aliasing?

Was sind also die gezackten Kanten, die durch Verlaufsgrafiken entstehen?

Eine einfache DEMO:

<div></div>
div {
    width: 500px;
    height: 100px;
    background: linear-gradient(37deg), #000 50%, #f00 50%, #f00 0);
}

Der Effekt ist wie folgt:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Tatsächlich ist das gezackte Gefühl bereits sehr deutlich zu erkennen und zu sehen, dass das Innere tatsächlich so ist:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Oder vielleicht so:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Interessanterweise ist das Aliasing-Phänomen besonders deutlich auf Bildschirmen mit einem DPR von 1, aber auf einigen hochauflösenden Bildschirmen (DPR > 1) ist das Gefühl nicht so offensichtlich.

DPR (Device Pixel Ratio) ist das Gerätepixelverhältnis, DPR = physische Pixel / geräteunabhängige Pixel. Das Gerätepixelverhältnis beschreibt das anfängliche proportionale Verhältnis zwischen physischen Pixeln und geräteunabhängigen Pixeln im unskalierten Zustand.

Warum gibt es also ein zackiges Gefühl?

Die Darstellung herkömmlicher Webseiten basiert auf Pixeleinheiten. Bei Bildern, bei denen eine Farbe direkt in einen anderen Farbzustand übergeht, kann es leicht zu einer Verschlechterung der visuellen Qualität (Informationsverzerrung) kommen. Daher tritt bei gewöhnlichen Verlaufselementen wie der oben genannten Schreibmethode Aliasing auf, was ein sehr häufiges heikles Problem bei der Verwendung von Farbverläufen darstellt.

Einfache Lösungen

Es gibt viele Lösungen für Verzerrungsprobleme. Der einfachste Weg besteht darin, keinen direkten Übergang vorzunehmen, sondern einen sehr kleinen Gradientenübergangsraum zu reservieren.

Wir können den obigen Code einfach umwandeln:

div {
    width: 500px;
    height: 100px;
  - background: linear-gradient(37deg), #000 50%, #f00 50%, #f00);
  + background: linear-gradient(37deg), #000 49.5%, #f00 50.5%, #f00);
}

Schauen Sie sich die Änderungen genau an. Wir haben von einem direkten Übergang von 50 % zu einem reservierten 1 %-Gradientenübergangsraum gewechselt. Der Effekt ist wie folgt:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Man sieht, dass sich die Wirkung sofort deutlich verbessert hat!

Wenn Sie den ursprünglichen Code nicht ändern möchten, können Sie dies natürlich auch durch Überlagern einer Ebene mit Pseudoelementen erreichen. Hier ist eine Vergleichstabelle der drei Methoden:

<div></div>
<div class="gradient"></div>
<div class="pesudo"></div>
:root {
    --deg: 37deg;
    --c1: #000;
    --c2: #f00;
    --line-width: 0.5px;
}
div {
    margin: auto;
    width: 500px;
    height: 100px;
    background: linear-gradient(
        var(--deg),
        var(--c1) 50%,
        var(--c2) 50%,
        var(--c2) 0
    );
}
// 方法一:
.gradient {
    background: linear-gradient(
        var(--deg),
        var(--c1),
        var(--c1) calc(50% - var(--line-width)),
        var(--c2) calc(50% + var(--line-width)),
        var(--c2) 0
    );
}
// 方法二:
.pesudo {
    position: relative;

    &::before {
        content: "";
        position: absolute;
        top: 0;
        left: 0;
        right: 0;
        bottom: 0;
        background: linear-gradient(
            var(--deg),
            transparent,
            transparent calc(50% - var(--line-width)),
            var(--c1) calc(50% - var(--line-width)),
            var(--c2) calc(50% + var(--line-width)),
            transparent calc(50% + var(--line-width)),
            transparent
        );
    }
}

Die Bedeutung der Überlagerung durch Pseudoelemente -elements soll Glätte dort erreichen, wo gezackte Kanten auftreten:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Der Effekt ist wie folgt:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

CodePen-Demo – Beseitigen Sie Farbverlaufsaliasing

Highlight! Diese Methode eignet sich für lineare Verläufe, radiale Verläufe und Winkelverläufe und ist die einfachste Möglichkeit, CSS-Aliasing zu eliminieren.

Fortgeschrittenere Methoden zur Aliasing-Beseitigung

Natürlich gibt es auch andere fortgeschrittenere Aliasing-Beseitigungsmethoden.

In diesem Artikel von Bionic LionCSS Anti-Aliasing wird auch eine weitere interessante Möglichkeit zur Eliminierung von Aliasing vorgestellt. Der folgende Inhalt ist teilweise ein Auszug aus dem Artikel.

Wir können eine Kantenalias-Kante-> Alias-Kanten rekonstruierende Methode erstellen.

Was wir tun müssen, ist, der gezackten Stelle eine weitere Inhaltsebene zu überlagern, um das gezackte Gefühl weniger intensiv zu machen. Es heißt Pixel-Offset-Anti-Aliasing (POAA).

Implementing FXAA这篇博客中,解释了 FXAA 具体是如何运作的。对于一个已经被找到的图形边缘,经过 FXAA 处理后会变成这样,见下两幅图:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

FXAA(Fast Approximate Anti-Aliasing),快速近似抗锯齿,它找到画面中所有图形的边缘并进行平滑处理。

我们可以轻易找到找到渐变的边缘地方,就是那些渐变的颜色改变的地方。有了边缘信息后,接着就要重建边缘。重建边缘也许可以再拆分,分为以下几个步骤:

  • 需要通过某种方法得到透明度的点
  • 这些点需要能够组成线段
  • 线段完全吻合我们的 Gradient
  • 使线段覆盖在 Gradient 的上一层以应用我们的修改

这就是大体思路,我们并没有参与浏览器的渲染,而是通过像 FXAA 一样的后处理的方法。在已渲染的图像上做文章。

比如说,我们有这样一张图:

.circle-con {
    $c1: #cd3f4f;
    $c2: #e6a964;
    position: relative;
    height: 300px;
    background-image: repeating-radial-gradient(
        circle at 0% 50%, 
        $c1 0, 
        $c2 50px
    );
}

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

边缘信息如下:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

我们要做的,就是在它的边缘处,利用渐变再生成一段渐变,通过准确叠加,消除渐变!原理图如下:

Detailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!

原理可行,但是实操起来非常之复杂,计算量会比较大。感兴趣的可以拿这段代码尝试一下:

.repeat-con {
    --c1: #cd3f4f;
    --c2: #e6a964;
    --c3: #5996cc;
    position: relative;
    height: 300px;
    background-image: repeating-linear-gradient(
        var(--deg),
        var(--c1),
        var(--c1) 10px,
        var(--c2) 10px,
        var(--c2) 40px,
        var(--c1) 40px,
        var(--c1) 50px,
        var(--c3) 50px,
        var(--c3) 80px
    );

    &.antialiasing {
        &:after {
            --offsetX: 0.4px;
            --offsetY: -0.1px;
            --dark-alpha: 0.3;
            --light-alpha: 0.6;
            --line-width: 0.6px;
            content: &#39;&#39;;
            position: absolute;
            top: var(--offsetY);
            left: var(--offsetX);
            width: 100%;
            height: 100%;
            opacity: 0.5;
            background-image: repeating-linear-gradient(
                var(--deg),
                var(--c3),
                transparent calc(0px + var(--line-width)),
                transparent calc(10px - var(--line-width)),
                var(--c2) 10px,
                var(--c1) 10px,
                transparent calc(10px + var(--line-width)),
                transparent calc(40px - var(--line-width)),
                var(--c1) 40px,
                var(--c2) 40px,
                transparent calc(40px + var(--line-width)),
                transparent calc(50px - var(--line-width)),
                var(--c3) 50px,
                var(--c1) 50px,
                transparent calc(50px + var(--line-width)),
                transparent calc(80px - var(--line-width)),
                var(--c1) 80px
            );
        }
    }
}

最后

简单总结一下,本文介绍了几种 CSS 中可行的消除渐变锯齿的方法。

好了,本文到此结束,希望本文对你有所帮助 :)

Das obige ist der detaillierte Inhalt vonDetailliertes Beispiel zur Lösung des Problems des CSS-Gradienten-Aliasings!. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:segmentfault.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen