Odtwarzanie filmów w jakości HDR

HDR, czyli High Dynamic Range, zapewnia szerszy zakres kolorów i większy kontrast między najjaśniejszymi bielami a najciemniejszymi cieniami, co przekłada się na jakość filmu, która bardziej przypomina to, co widzimy gołym okiem.

W aplikacji możesz skonfigurować odtwarzanie filmów HDR, aby wyświetlać podgląd i odtwarzać treści HDR.

W tym artykule zakładamy, że do Twojej aplikacji jest już dodana podstawowa obsługa odtwarzania wideo. Więcej informacji o odtwarzaniu znajdziesz w dokumentacji ExoPlayer.

Wymagania wstępne dotyczące urządzenia

Nie wszystkie urządzenia z Androidem obsługują odtwarzanie w HDR. Zanim odtworzysz treści wideo HDR w aplikacji, sprawdź, czy Twoje urządzenie spełnia te wymagania wstępne:

  • Jest kierowana na Androida w wersji 7.0 lub nowszej (w warstwie interfejsów API 24).
  • ma dekoder obsługujący HDR i dostęp do wyświetlacza obsługującego HDR;

Sprawdzanie obsługi odtwarzania w jakości HDR

Aby sprawdzić, czy wyświetlacz obsługuje HDR, użyj zapytania Display.getHdrCapabilities(). Metoda zwraca informacje o obsługiwanych profilach HDR i zakresie luminancji wyświetlacza.

Ten kod sprawdza, czy urządzenie obsługuje odtwarzanie HLG10. Począwszy od Androida 13 HLG10 to minimalny standard, który producenci muszą obsługiwać, jeśli urządzenia mogą odtwarzać treści HDR:

Kotlin

// Check if display supports the HDR type
val capabilities = display?.hdrCapabilities?.supportedHdrTypes ?: intArrayOf()
if (!capabilities.contains(HDR_TYPE_HLG)) {
  throw RuntimeException("Display does not support desired HDR type");
}

Java

// Check if display supports the HDR type
int[] list = getDisplay().getHdrCapabilities().getSupportedHdrTypes();
List capabilities = Arrays.stream(list).boxed().collect(Collectors.toList());
if (!capabilities.contains(HDR_TYPE_HLG)) {
 throw new RuntimeException("Display does not support desired HDR type");
}

Konfigurowanie odtwarzania w jakości HDR w aplikacji

Jeśli Twoja aplikacja korzysta z ExoPlayera, domyślnie obsługuje odtwarzanie w HDR. Dalsze kroki znajdziesz w artykule Sprawdzanie, czy funkcja odtwarzania HDR jest dostępna.

Jeśli Twoja aplikacja nie korzysta z ExoPlayera, skonfiguruj odtwarzanie HDR za pomocą MediaCodec przez SurfaceView.

Konfigurowanie MediaCodec za pomocą SurfaceView

Skonfiguruj standardowy proces odtwarzania MediaCodec za pomocą SurfaceView. To ustawienie pozwala na wyświetlanie filmów HDR bez specjalnej obsługi funkcji odtwarzania w tym trybie:

  • MediaCodec: dekoduje filmy HDR.
  • SurfaceView: wyświetla treści wideo HDR.

Ten kod sprawdza, czy kodek obsługuje profil HDR, a następnie konfiguruje MediaCodec za pomocą SurfaceView:

Kotlin

// Check if there's a codec that supports the specific HDR profile
val list = MediaCodecList(MediaCodecList.REGULAR_CODECS) var format = MediaFormat() /* media format from the container */;
format.setInteger(MediaFormat.KEY_PROFILE, MediaCodecInfo.CodecProfileLevel.AV1ProfileMain10)
val codecName = list.findDecoderForFormat (format) ?: throw RuntimeException ("No codec supports the format")

// Here is a standard MediaCodec playback flow
val codec: MediaCodec = MediaCodec.createByCodecName(codecName);
val surface: Surface = surfaceView.holder.surface
val callback: MediaCodec.Callback = (object : MediaCodec.Callback() {
   override fun onInputBufferAvailable(codec: MediaCodec, index: Int) {
      queue.offer(index)
   }

   override fun onOutputBufferAvailable(
      codec: MediaCodec,
      index: Int,
      info: MediaCodec.BufferInfo
   ) {
      codec.releaseOutputBuffer(index, timestamp)
   }

   override fun onError(codec: MediaCodec, e: MediaCodec.CodecException) {
      // handle error
   }

   override fun onOutputFormatChanged(
      codec: MediaCodec, format: MediaFormat
   ) {
      // handle format change
   }
})

codec.setCallback(callback)
codec.configure(format, surface, crypto, 0 /* flags */)
codec.start()
while (/* until EOS */) {
   val index = queue.poll()
   val buffer = codec.getInputBuffer(index)
   buffer?.put(/* write bitstream */)
   codec.queueInputBuffer(index, offset, size, timestamp, flags)
}
codec.stop()
codec.release()

Java

// Check if there's a codec that supports the specific HDR profile
MediaCodecList list = new MediaCodecList(MediaCodecList.REGULAR_CODECS);
MediaFormat format = /* media format from the container */;
format.setInteger(
    MediaFormat.KEY_PROFILE, CodecProfileLevel.AV1ProfileMain10);
String codecName = list.findDecoderForFormat(format);
if (codecName == null) {
    throw new RuntimeException("No codec supports the format");
}

// Below is a standard MediaCodec playback flow
MediaCodec codec = MediaCodec.getCodecByName(codecName);
Surface surface = surfaceView.getHolder().getSurface();
MediaCodec.Callback callback = new MediaCodec.Callback() {
    @Override
    void onInputBufferAvailable(MediaCodec codec, int index) {
        queue.offer(index);
    }

    @Override
    void onOutputBufferAvailable(MediaCodec codec, int index) {
        // release the buffer for render
        codec.releaseOutputBuffer(index, timestamp);
    }

    @Override
    void onOutputFormatChanged(MediaCodec codec, MediaFormat format) {
        // handle format change
    }

    @Override
    void onError(MediaCodec codec, MediaCodec.CodecException ex) {
        // handle error
    }

};
codec.setCallback(callback);
codec.configure(format, surface, crypto, 0 /* flags */);
codec.start();
while (/* until EOS */) {
    int index = queue.poll();
    ByteBuffer buffer = codec.getInputBuffer(index);
    buffer.put(/* write bitstream */);
    codec.queueInputBuffer(index, offset, size, timestamp, flags);
}
codec.stop();
codec.release();

Więcej implementacji interfejsu MediaCodec z wykorzystaniem SurfaceView znajdziesz w przykładach Aparatu z Androidem.

Materiały

Więcej informacji o odtwarzaniu HDR znajdziesz w tych materiałach:

HDR

Multimedia