Jetpack ピクチャー イン ピクチャー ライブラリを使用する

ピクチャー イン ピクチャー(PiP)Jetpack ライブラリは、Android アプリ デベロッパーが PiP 機能を実装するための効率的で堅牢なソリューションを提供します。特に、メディア再生、ビデオ通信、ナビゲーション アプリに適しています。このライブラリは、統一された API を提供することで、ボイラープレート コードやアプリ内での一般的なバグを排除し、PiP ユーザー エクスペリエンスの全体的な品質を向上させます。

PiP Jetpack ライブラリは、Android エコシステム全体にわたるいくつかの重要な課題と不整合に対処することで、既存の PiP API を容易にします。

  • OS の断片化: ライブラリは、Android 12 より前は enterPictureInPictureMode を使用し、Android 12 以降は isAutoEnterEnabled を使用するなど、さまざまな Android バージョン間の PiP API 呼び出しの違いを自動的に処理するため、デベロッパーはバージョンの違いを管理する必要がありません。
  • PiP パラメータの誤り: setSourceRectHint などの PiP パラメータを正しく設定して、メディア再生中にスムーズで高品質なアニメーションを作成するための統合ソリューションを提供します。
  • 統合された PiP 状態コールバック: onPictureInPictureModeChangedonPictureInPictureUiStateChanged を単一の統合コールバック インターフェース(PictureInPictureDelegate.OnPictureInPictureEventListener)に統合し、状態と UI の管理を簡素化します。
  • ボイラープレート コードの削減: 再生コントロールやビデオ通話アクションなどの一般的なユースケース向けに、事前定義された RemoteActions のセットを提供することで、繰り返し発生するボイラープレート コードの量を削減します。
  • 将来を見据えた設計: さらなる PiP 機能は Jetpack ライブラリを通じて提供されるため、導入者は最小限の労力で追加機能にアクセスできます。

Migration Workflow

アプリのユースケース カテゴリと以前の PiP ロジックを特定します。

カテゴリ: 動画再生、ナビゲーション、ビデオ通話。

以前の PiP ロジックで特定する:

  • onUserLeaveHint
  • setAutoEnterEnabled
  • onPictureInPictureModeChanged
  • onPictureInPictureUiStateChanged
  • setPictureInPictureParams

2. AndroidManifest の構成

PIP に移行するアクティビティが、不要な再起動を防ぐために必要な configChanges を使用して AndroidManifest.xml でサポートを宣言していることを確認します。

<activity
android:name="VideoActivity" android:supportsPictureInPicture="true"
android:configChanges="screenSize|smallestScreenSize|screenLayout|orientation">
</activity>

3. 環境設定

必要な依存関係を build.gradle に追加します。

dependencies {
implementation("androidx.core:core:1.18.0")
implementation("androidx.activity:activity:1.13.0")
implementation("androidx.core:core-pip:1.0.0-alpha02") }

依存関係には最新の AndroidX ライブラリを使用し、その情報についてはリリースのページを参照してください。

4. テンプレートの選択と初期化

アプリのユースケースに最も適した実装テンプレートを選択します。

  • ナビゲーションとビデオ通話: BasicPictureInPicture。シームレスなサイズ変更は通常サポートされておらず、ソースの矩形ヒントは必要ありません。
  • 動画再生: VideoPlaybackPictureInPicture。ソース rect ヒントのプレーヤー ビュー境界を自動的に追跡し、デフォルトでシームレスなサイズ変更を有効にします。

Jetpack ライブラリを採用するには、既存のカスタム PiP 実装を Jetpack ライブラリ API に置き換えます。採用の複雑さとコストは、アプリの現在の実装によって異なります。

以降のセクションでは、PiP の一般的なユースケースと、必要な実装手順について説明します。

アプリは、ナビゲーションの有効または無効の状態をライブラリに通知し、アスペクト比を設定します。残りの処理は Jetpack ライブラリが行います。

主な違い:

  1. アプリ側で自動入力と従来の入力を区別する必要はありません。
  2. 統合されたコールバック インターフェース。
  3. 下位互換性のための新しい PictureInPictureParams ビルダー。

ビデオ通話

アプリは、通話のアクティブ状態または非アクティブ状態をライブラリに通知し、アスペクト比を設定します。

主な違い:

  1. アプリ側で自動入力と従来の入力を区別する必要はありません。
  2. 統合されたコールバック インターフェース。
  3. 下位互換性のための新しい PictureInPictureParams ビルダー。
  4. ビデオ通話の標準化されたアクション アイコン。

5. コードの移行

  • エントリ ロジック: Android 12 以降の setAutoEnterEnabled や Android 11 以前の onUserLeaveHint などの API 固有のロジックを setEnabled に置き換えます。PiP の利用資格ステータスが変更されるたびに、これをトリガーします。
  • コールバック: onPictureInPictureModeChanged(レイアウトの切り替え)と onPictureInPictureUiStateChanged(アニメーション/状態)を統合して、イベントベースの統一コールバック onPictureInPictureEvent にします。
  • アクションとパラメータ: テンプレート インスタンスのパラメータが変更されるたびに、setActionssetAspectRatio を使用してパラメータを更新します。

リファレンス実装パターン

実装の例。

ナビゲーションとビデオ通話

class NavOrVideoCallJpipActivity : ComponentActivity(), PictureInPictureDelegate.OnPictureInPictureEventListener {
    private lateinit var pictureInPictureImpl: BasicPictureInPicture
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        pictureInPictureImpl = BasicPictureInPicture(this)
        // BasicPictureInPicture is ideal for Navigation and Video call use cases.
        pictureInPictureImpl.addOnPictureInPictureEventListener(
            ContextCompat.getMainExecutor(this),
            this
        )
        setContent {
        }
    }
    override fun onPictureInPictureEvent(
        event: PictureInPictureDelegate.Event,
        config: Configuration?
    ) {
        when (event) {
            PictureInPictureDelegate.Event.ENTERED -> { /* Toggle to PiP layout */ }
            PictureInPictureDelegate.Event.EXITED -> { /* Toggle to Full-screen layout */ }
            PictureInPictureDelegate.Event.STASHED -> { /* Optional: PiP is stashed */ }
            PictureInPictureDelegate.Event.UNSTASHED -> { /* Optional: PiP is unstashed */ }
        }
    }
}

動画再生

class VideoPlaybackJpipActivity : ComponentActivity(), PictureInPictureDelegate.OnPictureInPictureEventListener {
    private lateinit var pictureInPictureImpl: VideoPlaybackPictureInPicture
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        pictureInPictureImpl = VideoPlaybackPictureInPicture(this)
        pictureInPictureImpl.addOnPictureInPictureEventListener(
            ContextCompat.getMainExecutor(this),
            this
        )
        setContent {
            ContentScreen(pictureInPictureImpl)
        }
    }
    override fun onPictureInPictureEvent(
        event: PictureInPictureDelegate.Event,
        config: Configuration?
    ) {
        when (event) {
            PictureInPictureDelegate.Event.ENTER_ANIMATION_START -> { /* Hide overlays */ }
            PictureInPictureDelegate.Event.ENTER_ANIMATION_END -> { /* Animation finished */ }
            PictureInPictureDelegate.Event.ENTERED -> { /* Switch to PiP layout */ }
            PictureInPictureDelegate.Event.STASHED -> { /* PiP stashed */ }
            PictureInPictureDelegate.Event.UNSTASHED -> { /* PiP unstashed */ }
            PictureInPictureDelegate.Event.EXITED -> { /* Return to full-screen */ }
        }
    }

    @Composable
    fun ContentScreen(pipController: VideoPlaybackPictureInPicture) {
        DisposableEffect(pipController) {
            onDispose {
                pipController.close()
            }
        }
    }
}