TV에서 멀티태스킹

Android 14(API 수준 34)에서는 멀티태스킹을 허용하도록 PIP 모드 API를 일부 개선했습니다. PIP 지원은 Android 8.0 (API 수준 26)에서 도입되었지만 Android TV에서 널리 지원되지 않았으며 Android 13 이전의 Google TV에서는 전혀 지원되지 않았습니다. TV용 멀티태스킹은 PIP 모드를 사용하여 화면에 개별 앱 두 개가 공존할 수 있도록 합니다. 하나는 전체 화면으로 실행되고 다른 앱은 PIP 모드로 실행됩니다. 두 모드 중 하나에서 실행되는 앱에는 다른 요구사항이 있습니다.

기본 동작은 PIP 앱이 전체 화면 앱을 오버레이하는 것입니다. 이는 표준 Android PIP 모드 동작과 매우 동일합니다.

멀티태스킹을 통합할 때 애플리케이션은 TV 앱 품질 가이드라인에 따라 사용 유형을 선언해야 합니다.

PIP 모드로 앱 실행

Android 14 (API 수준 34) 이상을 실행하는 TV 기기의 경우 enterPictureInPictureMode()를 호출하여 PIP 모드로 앱을 실행합니다. 이전 버전의 Android를 실행하는 TV 기기는 PIP 모드를 지원하지 않습니다.

다음은 PIP 모드로 전환하는 버튼의 로직을 구현하는 방법의 예입니다.

Kotlin

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
    super.onViewCreated(view, savedInstanceState)
    pictureInPictureButton.visibility =
        if (requireActivity().packageManager.hasSystemFeature(FEATURE_PICTURE_IN_PICTURE)) {
            pictureInPictureButton.setOnClickListener {
                val aspectRatio = Rational(view.width, view.height)
                val params = PictureInPictureParams.Builder()
                    .setAspectRatio(aspectRatio)
                    .build()
                val result = requireActivity().enterPictureInPictureMode(params)
            }
            View.VISIBLE
        } else {
            View.GONE
        }
}

Java

@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);
    if (requireActivity().getPackageManager().hasSystemFeature(FEATURE_PICTURE_IN_PICTURE)) {
        pictureInPictureButton.setVisibility(View.VISIBLE);
        pictureInPictureButton.setOnClickListener(v -> {
            Rational aspectRatio = new Rational(view.getWidth(), view.getHeight());
            PictureInPictureParams params = new PictureInPictureParams.Builder()
                    .setAspectRatio(aspectRatio)
                    .setTitle("My Streaming App")
                    .setSubtitle("My On-Demand Content")
                    .build();
            Boolean result = requireActivity().enterPictureInPictureMode(params);
        });
    } else {
        pictureInPictureButton.setVisibility(View.GONE);
    }
}

이 작업은 기기에 시스템 기능 FEATURE_PICTURE_IN_PICTURE가 있는 경우에만 추가됩니다. 또한 작업이 트리거되면 PIP 모드의 가로세로 비율은 재생 중인 동영상의 가로세로 비율과 일치하도록 설정됩니다.

제목부제목을 추가하여 사용자에게 이 PIP의 일반적인 용도에 관한 정보를 제공해야 합니다.

PIP 모드에서 실행되는 앱과 공존

앱이 전체 화면 앱으로 실행되는 경우 PIP 모드로 실행되는 다른 앱에 맞게 조정해야 할 수 있습니다.

API-명확성

경우에 따라 PIP 앱이 전체 화면 앱 내에서 중요한 UI 구성요소를 오버레이할 수 있습니다. 이를 완화하기 위해 앱에서 오버레이해서는 안 되는 중요한 UI 구성요소를 식별하는 데 사용할 수 있는 명확한 API가 있습니다. 시스템은 PIP 창 위치를 변경하여 이러한 구성요소가 가려지지 않도록 요청을 수락하려고 합니다.

유지-지우기

뷰를 오버레이하지 않도록 지정하려면 다음 예와 같이 XML 레이아웃에서 preferKeepClear를 사용합니다.

<TextView
    android:id="@+id/important_text"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:preferKeepClear="true"
    android:text="@string/app_name"/>

setPreferKeepClear()를 사용하여 프로그래매틱 방식으로 이 작업을 실행할 수도 있습니다.

Kotlin

private lateinit var binding: MyLayoutBinding

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)

    binding = MyLayoutBinding.inflate(layoutInflater)
    setContentView(binding.root)
    binding.importantText.isPreferKeepClear = true
}

Java

private MyLayoutBinding binding;

@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    binding = MyLayoutBinding.inflate(getLayoutInflater());
    setContentView(binding.getRoot());
    binding.importantText.setPreferKeepClear(true);
}

View 전체를 삭제하지 않고 일부분만 명확하게 유지할 필요가 있는 경우가 있습니다. setPreferKeepClearRects()는 오버레이하면 안 되는 View의 영역을 지정하는 데 사용할 수 있습니다. Flutter, Jetpack Compose, WebView와 같이 View를 기본적으로 사용하지 않는 UI에는 영역을 명확하게 유지해야 하는 하위 섹션이 있을 수 있습니다. 이러한 경우에 이 API를 사용할 수 있습니다.

사용 유형

앱은 PIP 모드의 기본 사용 유형 또는 유형에 상응하는 com.google.android.tv.pip.category메타데이터 값 속성을 선언해야 합니다. android:supportsPictureInPicture="true"를 설정한 <activity>는 아래 표의 관련 값으로 이 속성을 선언해야 합니다.

이러한 카테고리에 속하지 않는 사용 유형, 특히 미디어 콘텐츠의 재생은 TV의 PIP 모드에서 허용되지 않습니다.

설명
"communication" 영상 통화 또는 음성 통화와 같은 통신 사용 사례
"smartHome" 스마트 홈 통합(예: 연결된 초인종 또는 베이비 모니터)
"health" 피트니스 추적 또는 건강 모니터링과 같은 건강 사용 사례
"ticker" 실시간 스포츠 점수 또는 뉴스 및 주식 티커와 같은 티커 사용 사례입니다.

값이 여러 개인 경우 세로 막대 (|)로 구분합니다. 예:

<meta-data android:name="com.google.android.tv.pip.category" android:value="smartHome|health" />