7
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

GLOBISAdvent Calendar 2023

Day 20

グロービス学び放題の字幕機能の秘密大公開!😄

Posted at

こんにちは、グロービスのフロントエンジニアAgataです。Udemyのような、動画学習サービスのグロービス学び放題というサイトの開発しております。

問題点

最近、ラーニングサポートのため、動画の隣に動画の全体の字幕が表示されるツールが増えてます。

例として、ユーチューブで使える「Language Learning with Netflix & YouTube-AFL」プラグインみてみましょう。

YouTubeの字幕プラグイン:紹介
⬆️「Language Learning with Netflix & YouTube-AFL」プラグインデモ
ソース: TED-Ed、How does the stock market work? - Oliver Elfenbaum)

動画再生とともに、右がわにある字幕が自動的にスクロールされます。
ツールをもう少し使ってみたら、一つ問題に気づくでしょう:
動画再生中、ユーザーがマウスなどで字幕をスクロールしても、自動で元の位置に戻ってしまう😅

YouTubeの字幕プラグイン:問題
⬆️ 自分でスクロールしても、再生中の字幕に戻ってしまう
ソース: TED-Ed、How does the stock market work? - Oliver Elfenbaum)

単純かのために、この記事では:

  • 「自動スクロール」: 字幕が自動的にスクロールされること
  • 「ユーザースクロール」:ユーザーが自分の意志で字幕をスクロールすること(マウスやタッチパッドなどで)

と呼びます。

グロービス学び放題では、このような字幕機能の実装に関わって、上記の問題を解決できました。解決までに辿り着く問題と解決仕方をこの記事で説明させていただきます。

修正アイデア

上記に説明した問題を解決するために、デザインをもらいました。
仕様は下記のようにでした:

  • 現在の状態:
    • 自動スクロール実装済み:動画の再生とともに、字幕が自動的にスクロールされました。
    • 実装方法:動画の時間がアップデートされるたびに、字幕が入っているdivをscrollToでスクロールさせました。
    • 問題:ユーザーが自分でスクロールしても、字幕の自動スクロールが止まってないこと
  • 解決仕様:
    • ユーザーが自分で字幕をスクロールすると、自動スクロールが止まって、「Resume Auto Scrolling」ボタンが表示されます Resume Auto Scrollingボタン

    • ユーザーが「Resume Auto Scrolling」ボタンを押すと、自動スクロールが再生されます。ボタンが消えます。

実装アイデア 1 と2

実装アイデア1

やり方:イベントリスナーで「自動スクロール」と「ユーザースクロール」を区別する
使えない理由:区別できない 💦

なぜ「自動スクロール」と「ユーザースクロール」を区別する必要があったかというと、
「自動スクロール」のスクロールイベントの場合は、自動スクロールを止めちゃいけなからです。
「ユーザースクロール」のスクロールイベントの場合だけ、自動スクロールを止める必要があります。

スクロールのイベントリスナー:

スクロール方法 onScroll onScroll以外
1. マウス 🟢 🟢(onWheel)
2. タッチパッド 🟢 🟢 (onTouchMove)
3. スクロールバー 🟢
4. 自動スクロール(scrollTo) 🟢

3つ目と4つ目の場合だと、スクロールイベントをonScrollでキャッチしかできないみたいです(間違えていたら、コメントで教えてください!)。
なので、3. スクロールバーのスクロールイベントと、4. 自動スクロールのスクロールイベントを区別できない状態です。

実装アイデア2

やり方:スクロールされた行数で「自動スクロール」と「ユーザースクロール」を区別する

  • 自動スクロール: 必ず1行〜3行のスクロール (それ以外の場合は、ユーザースクロールとする)1行

  • ユーザースクロール:4行以上にする (ユーザーが1行、100行でもスクロール可能です)4行

この情報を使って、下記のアルゴリズムを書いてみました。

アルゴリズム1

  • onScrollイベントリスナーを字幕が入っているdivに追加する。
    typescript <div onScroll={onScroll}> 字幕 </div>
  • onScrollのイベントリスナーでは、スクロールされた行数を確認する。
    • 4行以上の動きを確認できたら:
      • 自動スクロールをストップさせる
      • 「Resume Auto Scrolling」ボタンを表示する
    • 3行以下の動きを確認できたら:
      • 自動スクロールをストップさせない
  • ユーザーが「Resume Auto Scrolling」ボタンを押したら
    • 自動スクロールを開始する
    • 「Resume Auto Scrolling」ボタンを非表示される

具体的なコードを教えかねますが、上記の流れで実装してみたら、「自動スクロール」と「スクロールバー」の場合も区別できて、機能はほぼ完成の状態でした。

微調整

紹介したアルゴリズム1の懸念点は、ユーザーが3行以下スクロールしたら、自動スクロールが止まらないことでした。
マウスとタッチパッドでスクロールする場合だと、この状態を簡単に改善できます(実装アイデア1で紹介した「スクロールのイベントリスナー」一覧をご覧ください)。

下記のアルゴリズムを追加しました。

アルゴリズム2:

  • onWheelとonTouchMoveイベントリスナーを字幕が入っているdivに追加する。
<div
  onScroll={onScroll} // アルゴリズム1で追加したもの
  onWheel={onUserManualScroll}
  onTouchMove={onUserManualScroll}
>
  字幕
</div>
  • 上記のonUserManualScrollの定義:イベントリスナーが発火されたら、自動スクロールを停止し、「Resume Auto Scrolling」ボタンを表示する。

それのおかげで、ユーザーがマウスかタッチパッドで1行でもスクロールしても、すぐに「Resume Auto Scrolling」ボタンが表示されて、自動スクロールが止まります。

最後に

簡単そうに見えた機能でしたが、思ったより実装が面白かったです。

是非、グロービス学び放題でこの機能を使ってみてください!

7
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?