Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
CA BASE NEXT でスクロールに 連動したUIを構築した話
Search
Sponsored
·
Ship Features Fearlessly
Turn features on and off without deploys. Used by thousands of Ruby developers.
→
kubo-hide-kun
October 11, 2022
Programming
600
1
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
CA BASE NEXT でスクロールに 連動したUIを構築した話
kubo-hide-kun
October 11, 2022
More Decks by kubo-hide-kun
See All by kubo-hide-kun
ハイレベルな環境こそが最高である 科学的なお話
kubo_programmer
0
170
SQL Injection
kubo_programmer
0
130
IPアドレスとは何か?
kubo_programmer
0
3.7k
クライアント/サーバーシステム
kubo_programmer
0
15k
DHCPサーバ
kubo_programmer
0
3.4k
How to make Readable Slide
kubo_programmer
0
150
AtomicDesignの説明と所感
kubo_programmer
0
2k
Moonblock入門
kubo_programmer
3
1.4k
TCP/UDPの違い
kubo_programmer
4
6k
Other Decks in Programming
See All in Programming
Mujeres en SEO Summit 2026 - Greatest Disaster Hits en Web Performance
guaca
0
180
Inside Stream API
skrb
1
730
JavaDoc 再入門
nagise
1
360
軽量Java基盤の設計 DIコンテナに頼らない、長期保守と1秒起動の実現 JJUG CCC 2026 Spring
macha64
0
540
ふつうのFeature Flag実践入門
irof
7
4k
技術記事、 専門家としてのプログラマ、 言語化
mizchi
13
6.1k
Composerを使ったサプライチェーン攻撃の様子を眺めてみる #phpstudy
o0h
PRO
2
250
LLMによるContent Moderationの本番運用の裏側と品質担保への挑戦
suikabar
3
700
net-httpのHTTP/2対応について
naruse
0
490
脅威をエンジニアリングの糧にして――現場編 / Turning Threats into Engineering Fuel — Field Edition
nrslib
0
280
Even G2とAWSで推しのエージェントを召喚しよう!
har1101
1
120
並列実装の現場、2ヶ月間実務でAIを使い倒したAIもPCも私も限界が近い
ming_ayami
0
130
Featured
See All Featured
AI Search: Implications for SEO and How to Move Forward - #ShenzhenSEOConference
aleyda
1
1.3k
Designing Experiences People Love
moore
143
24k
Speed Design
sergeychernyshev
33
1.9k
How To Speak Unicorn (iThemes Webinar)
marktimemedia
1
490
Building an army of robots
kneath
306
46k
Navigating Algorithm Shifts & AI Overviews - #SMXNext
aleyda
1
1.3k
WCS-LA-2024
lcolladotor
0
640
Neural Spatial Audio Processing for Sound Field Analysis and Control
skoyamalab
0
330
個人開発の失敗を避けるイケてる考え方 / tips for indie hackers
panda_program
123
22k
Joys of Absence: A Defence of Solitary Play
codingconduct
1
390
Stop Working from a Prison Cell
hatefulcrawdad
274
21k
Site-Speed That Sticks
csswizardry
13
1.2k
Transcript
CA BASE NEXT でスクロールに 連動したUIを構築した話 ~ 発表者 : 窪田 ~
[名前] 窪田 秀哉 / クボ太郎 (2021年入社) [専門] React /
TypeScript [仕事] Marougeなどの複数の占いサービスを運用 [趣味] 知人にオススメされた漫画を読むこと。 今月、読み始めた作品: 「忘却バッテリー」「ハイパーインフレーション」「ドリフターズ」
1. CA BASE NEXT とは 2. スクロール連動UI a. 基本的な仕組み b.
安定して動かすための工夫 3. 告知
1. CA BASE NEXT とは 2. スクロール連動UI a. 基本的な仕組み b.
安定して動かすための工夫 3. 告知
“CA BASE NEXT” について
“CA BASE NEXT” (以下, ”CABN”) とは、 2022.7.27 ~ 2022.7.28 に開催に開催された
サイバーエージェントの 技術カンファレンス です。
自分は “LP開発チームのエンジニアメンバー” として CABNの運営に携わりました。 主に担当したのはスクロールに連動したUIの実装です。 「SANKOU!」「Web Clip Design」などの デザインまとめサイト にも掲載されました
🎉🎉
1. CA BASE NEXT とは 2. スクロール連動UI a. 基本的な仕組み b.
安定して動かすための工夫 3. 告知
【基本の仕組み】 パラパラ漫画 と同じ要領で、連続する複数枚の画像を高速で切り替えることで、 スクロールに連動して動画が動いてるように見せています。 CA BASE NEXT では ファーストビューで 141枚、エンドロールに
60枚 を使っています。
【パラパラ漫画機能を実現するために①】 このスクロール連動処理を行う上で大事になるのが、 スクロールの進捗状況 を インデックス番号(パラパラ漫画における何番目か) に変換する処理です. 例)進捗状況が全体の 10% であれば、 30枚目
の画像を表示する。 スクロール可能領域を基点としたブラウザの位置 = 表示領域のトップの位置を基点としたスクロール可能領域となるDOMの相対距離 * -1 window.addEventListener('scroll', () => { // ターゲットの上部から見た、スクロール量を取得 const positionTop = targetElm.getBoundingClientRect().top * -1; // スクロール可能領域の高さを取得 const scrollableHeight = targetElm.getBoundingClientRect().height; // スクロール可能領域を何%スクロールしたかを計算 const scrollFraction = positionTop / scrollableHeight; const frameIndex = Math.min( frameCount - 1, Math.floor(scrollFraction * frameCount) ); });
【パラパラ漫画機能を実現するために②】 取得したインデックス番号をもとに画像を描画します。 画像の描画は canvas の drawImage を使うことで実現します。 別案: ・videoタグの動画を進み具合にJSで操作する →
動画サイズが大きいと後半の内容の画質が悪くなる (再生しないとフレームをちゃんと読み込まない?) ・DOMを操作する → サイト自体がJSですごく重くなったので、見送り。 const img = await loadImage(framePaths[index]); const canvas = canvasRef.current; const context = canvas.getContext('2d'); // 中央寄せするための計算 (object-fix: cover; をJSで再現) const { offset, size } = calcCoverRect( { width: canvas.clientWidth, height: canvas.clientHeight }, { width: img.width, height: img.height } ); // 画質を落とさないための拡大率の計算 const scale = calcCanvasScale(canvas); requestAnimationFrame(() => { context.drawImage( img, offset.left * scale.x, offset.top * scale.y, size.width * scale.x, size.height * scale.y ); });
← 「青色」がウィンドウ。 「オレンジ」がスクロール可能領域。 「茶色」が表示されているDOM (ウィンドウ) これを駆使することで、
スクロールしてもパラパラ画面に相当するDOMを表示可能。 【パラパラ漫画機能を実現するために③】 次にスクロールしてもパラパラ漫画を表示し続けるためのCSSについて説明します。 やり方は表示したい要素(動画で言うと「茶色」のDOM)を画面目一杯に広げて、 position: fixed; もしくは position: sticky; をつけることで、 スクロールしても、特定の要素を表示し続けることができます。
基本の仕組みは先ほど説明した内容で十分ですが、 それだけだといくつか問題が発生するので、 その対応として自分が行った修正内容について解説していきます。
1. CA BASE NEXT とは 2. スクロール連動UI a. 基本的な仕組み b.
安定して動かすための工夫 3. 告知
【滑らかなアニメーションのための工夫】 素早くスクロールすると、画像フレームの切り替えに タイムラグ が生じてしまいます。 これは、新しい画像を表示する度に、画像のダウンロードを必要とするためです。 これを回避するために、スクロール前に あらかじめ画像をロード しておきます。 そうすれば、各フレームが既にダウンロードされてるので、 画像を滑らかにアニメーションすることができます。
const preloadImages = () => { currentFramePaths.forEach(loadImage); }; const loadImage = (src: string) => { return new Promise<HTMLImageElement>((resolve, reject) => { const img = new Image(); img.src = src; img.onload = () => resolve(img); img.onerror = () => reject(); }); };
【正常な描画ための工夫】 アクセスして数秒は画像の読み込みに時間がかかり、 パラパラ漫画の画像が何も表示されないという問題が発生しました。 これを解決するために最初の数フレームの画像は <link rel=”preload” … /> を使うことで、 HTMLが描画される時点で、最初の画像が読み込まれる状態を実現させる必要があります。
<NextHead> {alternateFrame.slice(0, 20).map((frame) => ( <React.Fragment key={frame.avif}> <link rel="preload" href={frame.avif} as="image" media="(min-width: 640px)" type="image/avif" /> </React.Fragment> ))} </NextHead>
【画像の軽量化のための工夫①】 パラパラ漫画のUIには必要となる画像枚数が多すぎるので、 avif , webP などの軽量な画像フォーマットにも対応しました。 html側であれば簡単に実現できますが、canvasで画像を表示しているので 、 JS側で「実行しているブラウザが各フォーマットに対応しているか」を確認するようにしています。 const
checkAvifSupport = (): Promise<boolean> => { return new Promise((resolve) => { const avif = new Image(); avif.src = 'data:image/avif;base64,...'; avif.onload = function () { const result = avif.width > 0 && avif.height > 0; resolve(result); }; avif.onerror = function () { resolve(false); }; }); }; const checkWebPSupport = (): Promise<boolean> => { return new Promise((resolve) => { const webP = new Image(); webP.src = 'data:image/webp;base64,...'; webP.onload = function () { const result = webP.width > 0 && webP.height > 0; resolve(result); }; webP.onerror = function () { resolve(false); }; }); }; ▼ webPが使えるか確認するメソッド ▼ avifが使えるか確認するメソッド この対応で jpg: 107MB → avif: 59MB (45%減)
【画像の軽量化のための工夫②】 スクロールの位置によって、画質の圧縮率を変更。 左の画像のように、背景が大きく写ってるタイミングは高画質。 右の画像のように、背景があまり描画されないタイミングは低画質にしています。 ▼ 高画質 ▼ 低画質
1. CA BASE NEXT とは 2. スクロール連動UI a. 基本的な仕組み b.
安定して動かすための工夫 3. 告知
CA BASE NEXT ではパラパラ漫画の機能を Reactで実現するために 600行 近く実装しています。 これをパラパラ漫画を実現したいと思う人が毎回書くのはツラいすぎるので、 今回、開発したロジックを OSS
として公開することにしました。 (会社から許可はもらっていますが、あくまで 個人名義のライブラリ です)
既にpublish済みですが、READMEが未整備です。 (進展があったら Twitter で告知するので、ぜひフォローお願いします🙏 → 今回のイベントのCompass から飛べます) デモサイトも開発中です。 (こちらも完成したら Twitter
で告知します) 【開発中のライブラリ】 ライブラリ①: スクロールの進捗状況を計算するカスタムフック。 ライブラリ②: 画像を渡すだけでパラパラ漫画機能を実現可能なコンポーネント。 window.addEventListener('scroll', () => { // ターゲットの上部から見た、スクロール量を取得 const positionTop = targetElm.getBoundingClientRect().top * -1; // スクロール可能領域の高さを取得 const scrollableHeight = targetElm.getBoundingClientRect().height; // スクロール可能領域を何%スクロールしたかを計算 const scrollFraction = positionTop / scrollableHeight; }); ▼ カスタムフックで提供するロジック
発表は以上です。 最後までお聞きいただき、ありがとうございました。 Presentation by クボ太郎 ( Twitter: @kubo_programmer )