目次
概要 " > 概要
视差的问题分析
不好的方案:使用滚动事件
不好的方案:更新背景位置
CSS 3D
调整perspective的比例
悪い。解決策: スクロール イベントを使用する
悪い解決策: 背景の位置を更新する
遠近感の比率を調整します
为什么这种方案好用?
美中不足:Mobile Safari
固定定位的问题
花式bug和应对
结论
ホームページ ウェブフロントエンド CSSチュートリアル 高性能視差アニメーション【翻訳】

高性能視差アニメーション【翻訳】

Apr 04, 2017 am 10:21 AM

高性能視差アニメーション


好き嫌いは関係なく、視差効果は賢く使用すればウェブ上で広く知られていますアプリケーションに奥行きと比喩的な効果を追加できますが、問題は、高性能の視差効果を実装することが非常に困難な作業であることです。もちろん、この記事では、視差効果のパフォーマンスを構築する方法について説明します。 、クロスブラウザでもあります

高性能視差アニメーション【翻訳】

視差効果図

概要

  • 視差アニメーションを作成するために、スクロールイベント (scroll events) や背景の位置決め (background -position) を使用しないでください。

  • CSS 3D 変換を使用して、より正確な視差効果を作成します

  • iOS モバイル デバイスの Safari ブラウザの場合は、視差が機能することを確認してください

    position: sticky 来确保视差可以生效。

如果你想要一个开箱即用的方案,请访问 Parallax helper JS ,这里还有一个 demo演示。

视差的问题分析

在开始之前,我们先来看两个现有的常见的实现视差的方法,探讨为何它们不适合我们要追求的目标。

不好的方案:使用滚动事件

视差的关键需求是它应该是滚动耦合的:对于页面滚动的每一个位置变化,视差元素的位置也应被更新。这看上去很容易,现代浏览器的重要机制之一就是它们可以异步处理工作。尽管如此,在大多数浏览器中,滚动事件是被作为“尽量好”(best effort)的工作处理的,也就意味着:浏览器并不确保每一帧的滚动动画送达!

这个重要的信息告诉我们为什么要避免使用Javascript基于滚动事件去移动元素:Javascript并不能确保视差会和页面滚动保持同样的步调。。在一些老版本的Mobile Safari上,滚动事件甚至是在滚动完成后才触发的,这一点让基于Javascript的滚动效果无法实现。在较新的Mobile Safari版本中,滚动事件可以在动画过程中触发了,但是和Chrome一样,它是一个基于“尽量好”的原则的。所以当主线程忙于其他工作时,滚动事件不会立即触发,也就意味着视差效果的丢失。

不好的方案:更新背景位置

我们要避免的另一个场景是在每帧都进行绘制。很多方案试图采用改变 background-position 来提供视差效果, 但这会让浏览器在滚动时重绘受影响的部分,而这可能会是相当消耗资源的,这种影响会使动画卡顿。

如果我们想提供一个高质量的视差动画,我们想要的是一个可以当作加速的属性(这里我们指的是 transformopacity ),而这是不依赖于滚动事件的。

CSS 3D

Scott Kellum 和 Keith Clark 都已经在利用 CSS 3D 来实现视差效果领域做出了很重要的工作。他们采用的非常有效技术有:

  1. 建立一个容器元素,设置 <a href="http://www.php.cn/wiki/926.html" target="_blank">overflow-y</a>: scroll 使其可以滚动(同时可能需要 <a href="http://www.php.cn/wiki/924.html" target="_blank">overflow-x</a>: hidden)。

  2. 对于上面的元素, 我们会应用一个 perspective 值,然后设置 perspective-origin<a href="http://www.php.cn/wiki/904.html" target="_blank">top</a> left, 或者 0 0

  3. 对上面元素的子元素应用一个在 Z 轴的变换,然后把它们还原回来以实现视差效果,而没有影响它们在屏幕上的大小。

这种方案的 CSS 看起来是下面的样子:

.container {
  width: 100%;
  height: 100%;
  overflow-x: hidden;
  overflow-y: scroll;
  perspective: 1px;
  perspective-origin: 0 0;
}

.parallax-child {
  transform-origin: 0 0;
  transform: translateZ(-2px) scale(3);
}
ログイン後にコピー

当然我们假定你的 HTML 是下面的样子:

<p class="container”>
  <p class="parallax-child”></p>
</p>
ログイン後にコピー

调整perspective的比例

把子元素挤回来会要求它设置一个更小的相对于 perspective。すぐに使えるソリューションが必要な場合は、視差ヘルパー JS にアクセスしてください。デモのデモンストレーションも行います。 視差の問題の分析

🎜 始める前に、視差を実装する 2 つの一般的な方法を見て、それらが私たちが追求する目標に適していない理由を説明します。🎜

悪い。解決策: スクロール イベントを使用する

🎜視差の重要な要件は、視差がスクロールと連動していることです。ページがスクロールするときに位置が変化するたびに、視差要素の位置も 更新 🎜 簡単そうに思えますが、最近のブラウザの重要なメカニズムの 1 つは非同期処理ができることです。ただし、ほとんどのブラウザでは非同期処理が可能です。 、スクロール イベントは「ベスト エフォート」ジョブとして処理されます。つまり、ブラウザはすべてのフレームでスクロール アニメーションを実行することを保証しません。この重要な情報は、JavaScript🎜 を使用して、スクロール イベントに基づいて要素を移動します。 🎜 Javascript では、視差がページのスクロールと同じペースを維持することが保証されません。 🎜。一部の古いバージョンではMobile Safari では、スクロール イベントはスクロールの完了後にもトリガーされるため、JavaScript に基づくスクロール効果は不可能になります。ただし、Chr🎜 ome と同様に、「可能な限り良いもの」の原則に基づいています。そのため、メインスレッドが他の作業でビジー状態の場合、スクロール イベントはすぐにはトリガーされず、視差効果が失われます。 🎜

悪い解決策: 背景の位置を更新する

🎜 避けたいもう 1 つのシナリオは、フレームごとに描画することです。多くのソリューションは、background-position を変更することで視差効果を提供しようとしますが、これによりスクロール時にブラウザが影響を受ける部分を再描画することになり、この効果によりアニメーションが途切れる原因となる可能性があります。 。 🎜🎜高品質の視差アニメーションを提供したい場合、必要なのは高速化された
プロパティ 🎜 (ここでは、スクロール イベントに依存しない transformopacity を参照します。 🎜🎜CSS 3D🎜🎜Scott Kellum と Keith Clark は、CSS 3D を使用して視差効果を実現する分野で重要な研究を行っています。彼らが使用する非常に効果的なテクニックは次のとおりです: 🎜
    🎜🎜 コンテナ要素を作成し、<a href="http://www.php.cn/wiki/926%5D%20%E3%82%92%E8%A8%AD%E5%AE%9A%E3%81%97%E3%81%BE%E3%81%99%E3%80%82%20.html" target="_blank">overflow-y🎜: スクロール</a> してスクロール可能にします (<a href="http://www.php.cn/wiki/924.html%20%E3%82%82%E5%BF%85%E8%A6%81%E3%81%AA%E5%A0%B4%E5%90%88%E3%81%8C%E3%81%82%E3%82%8A%E3%81%BE%E3%81%99)%20" target="_blank">overflow-x🎜: 非表示</a>)。 🎜🎜🎜🎜上記の要素では、perspective 値を適用し、perspective-origin<a href="http://www" .php.cn target="_blank">左上</a>、または 0 0。 🎜🎜🎜🎜上記の要素の子要素に Z 軸変換を適用し、それらを復元して、画面上のサイズに影響を与えることなく視差効果を実現します。 🎜🎜
🎜このソリューションの CSS は次のようになります:🎜
<p class="container”>
  <p class="parallax-container”>
    <p class="parallax-child”></p>
  </p>
</p>
ログイン後にコピー
ログイン後にコピー
🎜もちろん、HTML は次のようになると仮定します:🎜
.parallax-container {
  transform-style: preserve-3d;
}
ログイン後にコピー
ログイン後にコピー

遠近感の比率を調整します

🎜子の配置要素を押し戻すには、perspective に対して相対的に小さい比率を設定する必要があります。必要なズーム率は次の式で計算できます: 🎜🎜(視点 - 距離) / 視点🎜🎜。視差要素を最初に設定したのと同じくらい大きく表示したいので、同じままではなく、この方程式に従ってスケールする必要があります。 🎜

拿上面的例子来说, perspective1px, 而 parallax-child 在 Z 轴方向是 -2px,这就意味着元素需要被放大3倍,你可以看到我们在 scale 中写入了 3 这个值。

对于任何没有应用 translateZ 的内容,你可以用 0 替代,也就是缩放比为 (perspective - 0) / perspective,结果为 1 ,即既不放大也不缩小。真的是非常方便。

为什么这种方案好用?

弄清楚为什么这种方案好用是非常重要的,因为我们很快就要使用这个知识了。滚动其实本质是一种变换,这是它为什么可以被加速的原因。滚动很大程度上使用GPU参与了图层的变换。一个常见的滚动(没有应用任何 perspective )是这样的:滚动这种情况下是以 1:1 的方式在对待滚动的元素和它的子元素。换人话说,如果你向下滚动一个元素 <a href="http://www.php.cn/code/4221.html" target="_blank">300</a>px,那么它的子元素向上变换了同样的数量: 300px

但是,如果对这个滚动元素应用 perspective 值会把这个过程“搞乱”:这个值改变了滚动变换的理想路线。现在如果一个 300px 的滚动可能把子元素移动了 150px,当然这取决于你给 perspectivetranslateZ 设置什么值。如果一个 translateZ 设置为0的子元素,它的滚动会一切如常 ( 1:1 ),但是一个被推向 Z 轴向( translateZ 不为 0 )的子元素将以不同的比例滚动!因此出现了视差效果。另外非常重要的一点是:这个过程本身就是浏览器内部的滚动机制的一部分,因此没有必要监听滚动事件或者改变背景位置。

美中不足:Mobile Safari

每种效果都有一些约束,对于变换( transform )来讲,对子元素的 3D 效果的保持就是这样。如果在应用 perspective 的元素和它的“视差”子元素的结构之中有其它元素的存在,那么 3D 的效果会被“拍扁”,也就是说效果将不复存在。

<p class="container”>
  <p class="parallax-container”>
    <p class="parallax-child”></p>
  </p>
</p>
ログイン後にコピー
ログイン後にコピー

在上面的HTML中 .parallax-container 是一个新添加的元素,而它会"抹平" perspective,从而导致效果丢失。通常情况下,解决方案还是比较符合直觉的:给这个元素添加 transform-style: preserve-3d 以便让它可以把 3D 效果应用到更深层的节点去。

.parallax-container {
  transform-style: preserve-3d;
}
ログイン後にコピー
ログイン後にコピー

对于 Mobile Safari 来说,事情变的有点麻烦。对容器元素应用  overflow-y : 技术上这是没问题的,但是这会让滚动元素的移动过于凶猛。解决方案是加上一个  -webkit-overflow-scrolling: touch ,但这个设置会导致 perspective 被抹平,因此我们会得不到任何视差效果。

从一个发展的角度看,这可能算不上什么问题(因为只在旧版本的 Mobile Safari 出现),即使我们无法在每一个浏览器中展现视差效果,但一个应用的功能还是好用的,但我们最好找出一个方案。

position:sticky 来拯救

事实上,我们可以从 position: sticky 中得到一些帮助,这个设置允许元素固定在 <a href="http://www.php.cn/css/css-rwd-viewport.html" target="_blank">viewport</a> 的顶部或者固定在一个滚动元素的父元素。这个属性的文档,就如同任何其它文档一样,又臭又长,但是还是可以找到一些有用信息:

一个固定的“盒子”非常像一个相对定位的盒子,但是位移是通过引用最近的可滚动的祖先来计算的,或者根据 viewport 来计算,如果找不到这样一个祖先的话 -- CSS Positioned Layout Module Level 3

第一眼看上去好像帮助不大,但一个关键点在句中说到如何计算元素的固定位置的那部分:“位移是通过引用最近的可滚动的祖先来计算的”。换句话说,移动固定元素的距离(为了让它看起来是固定在某个元素或者 viewport 上)是在应用其它任何变换之前进行计算的,而不是之后。这就意味着,和我们刚才说的滚动的那个例子很像,如果位移计算的结果是 300px,那么我们就有了一个新的机会去使用 perspective (或者其它任何变换)来在这个 300px 应用到固定元素之前去改变它。

通过给视差元素设置 position: -webkit-sticky,我们可以有效的“翻转”那个由于 -webkit-overflow-scrolling: touch 而产生的“抹平”效果。这样就确保了视差元素引用最近的可滚动的祖先元素,这里就是 .container 。然后,和上文讲的类似,给 .parallax-container 设置一个 perspective 值,这样就改变了计算的滚动位移,创建出了视差效果。

<p class="container”>
  <p class="parallax-container”>
    <p class="parallax-child”></p>
  </p>
</p>

.container {
  overflow-y: scroll;
  -webkit-overflow-scrolling: touch;
}

.parallax-container {
  perspective: 1px;
}

.parallax-child {
  position: -webkit-sticky;
  top: 0px;
  transform: translate(-2px) scale(3);
}
ログイン後にコピー

这样就为 Mobile Safari 恢复了视差效果,真是一个不错的结果。

固定定位的问题

和之前的方案确实还有一个明显区别, position: sticky 改变了视差的机制。固定定位试图固定某个元素在滚动容器顶端,而非固定元素不是。这就意味着固定定位产生的视差和非固定产生的色差是相反的:

  • 使用 position: sticky: 元素离 z=0 越近,它移动的越少

  • 不使用 position: sticky: 元素离 z=0 越近,它移动的越多

如果你还是感到有些抽象的话,可以看一下Robert Flack的这个demo,这个demo展示了在固定定位和非固定定位的条件下,元素是如何有不同的表现的。要看到这个效果的话,你需要 Chrome Canary (写作本文是,版本为56) 或者 Safari 。

高性能視差アニメーション【翻訳】

position: sticky 对视差的影响

花式bug和应对

如同任何事情一样,还是有很多的坑需要填。

  • 固定定位的支持是不一致的:当前在 Chrome 中对于这个特性还在开发中,Edge 则完全缺失,FireFox则有绘制的bug。在这种情况下,我们应该增加一点代码来在需要时(这里就是 Mobile Safari )才添加 position: sticky

  • 该效果在 Edge 中完全没有作用。Edge试图在OS级别处理滚动,通常情况下这是个好事。但在这个例子中,这种机制会使得我们无法在滚动时去应用 perspective。为了修复这个问题,我们可以为父元素 设置 `translateZ(0px)``。

  • 页面内容太大了:在决定页面内容有多大时,很多浏览器负责放缩,但很遗憾 Chrome 和 Safari 不负责。所以假如有一个放大 3 倍的变换应用到某个元素时,你可能会看到滚动条出现了,而且一旦出现后,即使之后你恢复了 1:1 的比例,滚动条也不会消失。有一个方法可以避免这种情况:那就是从右下角进行放缩( transform-origin: <a href="http://www.php.cn/wiki/906.html" target="_blank">bottom</a> <a href="http://www.php.cn/wiki/905.html" target="_blank">right</a> )。这种方案背后的原理是它会导致过大的元素进入滚动区域的“负面”(一般是左上),而滚动区域永远不会让你看到“负面”区域。

结论

视差动画如果经过的周全的设计考虑后会是一个非常有趣的效果。而且现在你应该可以了解到我们是可以实现一个高性能的、滚动耦合的、跨浏览器的方案。由于这里面需要一点点数学计算和一些模板化的操作,所以我们封装了一个工具类和例子。

欢迎试用,并提出您的宝贵意见。

以上が高性能視差アニメーション【翻訳】の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

Swoole を使用して高性能 HTTP リバース プロキシ サーバーを実装する方法 Swoole を使用して高性能 HTTP リバース プロキシ サーバーを実装する方法 Nov 07, 2023 am 08:18 AM

Swoole を使用して高性能 HTTP リバース プロキシ サーバーを実装する方法 Swoole は、PHP 言語に基づいた高性能、非同期、同時ネットワーク通信フレームワークです。一連のネットワーク機能を提供し、HTTP サーバー、WebSocket サーバーなどの実装に使用できます。この記事では、Swoole を使用して高性能 HTTP リバース プロキシ サーバーを実装する方法と、具体的なコード例を紹介します。環境構成 まず、サーバーに Swoole 拡張機能をインストールする必要があります

PHP と WebSocket: 高性能のリアルタイム アプリケーションの構築 PHP と WebSocket: 高性能のリアルタイム アプリケーションの構築 Dec 17, 2023 pm 12:58 PM

PHP と WebSocket: 高性能リアルタイム アプリケーションの構築 インターネットが発展し、ユーザーのニーズが高まるにつれて、リアルタイム アプリケーションはますます一般的になってきています。従来の HTTP プロトコルには、最新のデータを取得するために頻繁なポーリングや長時間のポーリングが必要になるなど、リアルタイム データを処理する場合にいくつかの制限があります。この問題を解決するために、WebSocket が登場しました。 WebSocket は、双方向通信機能を提供する高度な通信プロトコルであり、ブラウザとサーバーの間でリアルタイムの送受信を可能にします。

C++ ハイパフォーマンス プログラミングのヒント: 大規模なデータ処理のためのコードの最適化 C++ ハイパフォーマンス プログラミングのヒント: 大規模なデータ処理のためのコードの最適化 Nov 27, 2023 am 08:29 AM

C++ は、開発者に柔軟性と拡張性を提供する高性能プログラミング言語です。特に大規模なデータ処理シナリオでは、C++ の効率と高速な計算速度が非常に重要です。この記事では、大規模なデータ処理のニーズに対応するために C++ コードを最適化するためのテクニックをいくつか紹介します。従来の配列の代わりに STL コンテナを使用する C++ プログラミングでは、配列は一般的に使用されるデータ構造の 1 つです。ただし、大規模なデータ処理では、vector、deque、list、set などの STL コンテナーを使用すると、より多くの処理が実行される可能性があります。

Go 言語を使用して高性能音声認識アプリケーションを開発および実装する Go 言語を使用して高性能音声認識アプリケーションを開発および実装する Nov 20, 2023 am 08:11 AM

科学技術の継続的な発展に伴い、音声認識技術も大きく進歩し、応用されています。音声認識アプリケーションは、音声アシスタント、スマート スピーカー、仮想現実などの分野で広く使用されており、より便利でインテリジェントな対話方法を人々に提供しています。高性能音声認識アプリケーションを実装する方法は、検討する価値のある問題となっています。近年、音声認識アプリケーションの開発において、高機能なプログラミング言語としてGo言語が注目を集めています。 Go 言語は、同時実行性が高く、記述が簡潔で、実行速度が速いという特徴があり、高パフォーマンスの言語を構築するのに非常に適しています。

Go 言語を使用して高性能の顔認識アプリケーションを開発する Go 言語を使用して高性能の顔認識アプリケーションを開発する Nov 20, 2023 am 09:48 AM

Go 言語を使用して高性能の顔認識アプリケーションを開発する 要約: 顔認識テクノロジは、今日のインターネット時代において非常に人気のあるアプリケーション分野です。この記事では、Go 言語を使用して高性能の顔認識アプリケーションを開発する手順とプロセスを紹介します。 Go 言語の同時実行性、高性能、使いやすさの機能を使用することで、開発者は高性能の顔認識アプリケーションをより簡単に構築できます。はじめに: 今日の情報社会では、顔認識技術はセキュリティ監視、顔決済、顔ロック解除などの分野で広く使用されています。インターネットの急速な発展により、

Java 開発: Netty を使用して高性能ネットワーク プログラミングを行う方法 Java 開発: Netty を使用して高性能ネットワーク プログラミングを行う方法 Sep 20, 2023 pm 02:09 PM

Java 開発: Netty を使用して高性能ネットワーク プログラミングを行う方法 概要: Netty は、ネットワーク アプリケーションの開発プロセスを簡素化する、高性能の非同期イベント駆動型ネットワーク プログラミング フレームワークです。この記事では、Netty の主な機能と、Netty を使用して高性能ネットワーク プログラミングを行う方法を紹介します。同時に、読者が Netty をよりよく理解し、適用できるように、いくつかの具体的な Java コード例も提供します。 1. Netty の概要 Netty は JavaNIO に基づいたネットワーク プログラミング ボックスです。

Docker と Spring Boot の技術的実践: 高パフォーマンスのアプリケーション サービスを迅速に構築する Docker と Spring Boot の技術的実践: 高パフォーマンスのアプリケーション サービスを迅速に構築する Oct 21, 2023 am 08:18 AM

Docker と SpringBoot の技術的実践: 高パフォーマンスのアプリケーション サービスを迅速に構築する はじめに: 今日の情報化時代において、インターネット アプリケーションの開発と展開はますます重要になっています。クラウド コンピューティングと仮想化テクノロジの急速な発展に伴い、軽量コンテナ テクノロジとして Docker が広く注目され、応用されるようになりました。 SpringBoot は、Java アプリケーションを迅速に開発および展開するためのフレームワークとしても広く認識されています。この記事では、Docker と SpringB を組み合わせる方法について説明します。

高性能データベース検索アルゴリズムのための Java 実装のアイデア 高性能データベース検索アルゴリズムのための Java 実装のアイデア Sep 18, 2023 pm 01:39 PM

高性能データベース検索アルゴリズムのための Java 実装アイデアの要約: インターネットとビッグデータ時代の到来により、データベースのストレージと検索パフォーマンスはデータ処理の効率にとって非常に重要です。この記事では、高性能データベース検索アルゴリズムの Java 実装アイデアを紹介し、具体的なコード例を示します。はじめに データベース検索は、大規模なデータ収集で高速にクエリを実行するための重要な操作の 1 つです。従来のデータベース検索アルゴリズムでは検索効率が低いという問題があり、ビッグデータ時代のニーズに応えられません。したがって、高性能のデータベース検索アルゴリズムは、

See all articles