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
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.3k
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
技術記事、 専門家としてのプログラマ、 言語化
mizchi
4
2.7k
Spring Security 実践 ─ GraphQL APIで実務に役立つ 認証・認可 を学ぶ
wagyu
0
220
運用エージェントは "作る" から "育てる" へ - 記憶と自己進化の3層設計パターン / self-evolving-agents-three-layer-agent-design
gawa
12
3.6k
過去最大のMCPアップデート! 2026-07-28 RC版の謎に迫る
licux
6
240
ローカルLLMを使ってB2Bサービスを作っていての学び
yaotti
0
160
決定論的オーケストレーションの設計と実装 / Design and Implementation of Deterministic Orchestration
nrslib
3
1.3k
The ROI of Quarkus for Spring Boot Applications
hollycummins
0
110
AI 時代のソフトウェア設計の学び方
masuda220
PRO
29
12k
並列実装の現場、2ヶ月間実務でAIを使い倒したAIもPCも私も限界が近い
ming_ayami
0
120
Semantic Version 単位で戦略を柔軟に変えて、パッケージアップデートを自動化する
daitasu
0
220
The NotImplementedError Problem in Ruby
koic
1
710
コンテキストの使い捨てをやめる — ビジネスルール駆動開発と miko —
ioki
0
190
Featured
See All Featured
How Fast Is Fast Enough? [PerfNow 2025]
tammyeverts
3
610
SEOcharity - Dark patterns in SEO and UX: How to avoid them and build a more ethical web
sarafernandez
0
200
Imperfection Machines: The Place of Print at Facebook
scottboms
270
14k
Facilitating Awesome Meetings
lara
57
7k
世界の人気アプリ100個を分析して見えたペイウォール設計の心得
akihiro_kokubo
PRO
71
40k
Odyssey Design
rkendrick25
PRO
2
690
Ten Tips & Tricks for a 🌱 transition
stuffmc
0
130
Reflections from 52 weeks, 52 projects
jeffersonlam
356
21k
Sharpening the Axe: The Primacy of Toolmaking
bcantrill
46
2.9k
What the history of the web can teach us about the future of AI
inesmontani
PRO
1
610
The Anti-SEO Checklist Checklist. Pubcon Cyber Week
ryanjones
0
160
The Success of Rails: Ensuring Growth for the Next 100 Years
eileencodes
47
8.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 )