Aplicación de demostración de ExoPlayer

La app de demostración principal de ExoPlayer tiene dos propósitos principales:

  1. Proporcionar un ejemplo relativamente simple pero con todas las funciones del uso de ExoPlayer La app de demostración se puede usar como un punto de partida conveniente para desarrollar tu propia app.
  2. Para facilitar la prueba de ExoPlayer. Puedes usar la app de demostración para probar la reproducción de tu propio contenido además de las muestras incluidas.

En esta página, se describe cómo obtener, compilar y ejecutar la app de demostración. También se describe cómo usarla para reproducir tu propio contenido multimedia.

Obtén el código

Puedes encontrar el código fuente de la app de demostración principal en la carpeta demos/main de nuestro proyecto de GitHub. Si aún no lo hiciste, clona el proyecto en un directorio local:

git clone https://github.com/androidx/media.git

A continuación, abre el proyecto en Android Studio. Deberías ver lo siguiente en la vista Android Project (se expandieron las carpetas relevantes de la app de demostración):

El proyecto en Android Studio

Compila y ejecuta

Para compilar y ejecutar la app de demostración, selecciona y ejecuta la configuración demo en Android Studio. Se instalará y ejecutará la app de demostración en un dispositivo Android conectado. Te recomendamos que uses un dispositivo físico si es posible. Si deseas usar un emulador, lee la sección sobre emuladores en Dispositivos compatibles y asegúrate de que tu dispositivo virtual use una imagen del sistema con un nivel de API de al menos 23.

SampleChooserActivity y PlayerActivity

La app de demostración presenta una lista de muestras (SampleChooserActivity). Si seleccionas una muestra, se abrirá una segunda actividad (PlayerActivity) para reproducir. La demostración incluye controles de reproducción y la funcionalidad de selección de pistas. También usa la clase de utilidad EventLogger de ExoPlayer para enviar información de depuración útil al registro del sistema. Este registro se puede ver (junto con el registro de nivel de error para otras etiquetas) con el siguiente comando:

adb logcat EventLogger:V *:E

Habilita decodificadores agrupados

ExoPlayer tiene varias extensiones que permiten el uso de decodificadores de software en paquetes, incluidos AV1, VP9, Opus, FLAC y FFmpeg (solo audio). La app de demostración se puede compilar para incluir y usar estas extensiones de la siguiente manera:

  1. Crea cada una de las extensiones que quieras incluir. Ten en cuenta que este es un proceso manual. Consulta el archivo README.md en cada extensión para obtener instrucciones.
  2. En la vista Build Variants de Android Studio, establece la variante de compilación del módulo de demostración en withDecoderExtensionsDebug o withDecoderExtensionsRelease, como se muestra en la siguiente imagen.

    Cómo seleccionar la variante de compilación de demostración `withDecoderExtensionsDebug`

  3. Compila, instala y ejecuta la configuración demo con normalidad.

De forma predeterminada, solo se usará un decodificador de extensiones si no existe un decodificador de plataforma adecuado. Es posible especificar la preferencia de los decodificadores de extensiones, como se describe en las siguientes secciones.

Cómo reproducir tu propio contenido

Existen varias formas de reproducir tu propio contenido en la app de demostración.

1. Edición de assets/media.exolist.json

Las muestras enumeradas en la app de demostración se cargan desde assets/media.exolist.json. Si editas este archivo JSON, es posible agregar y quitar muestras de la app de demostración. El esquema es el siguiente, donde [O] indica un atributo opcional.

[
  {
    "name": "Name of heading",
    "samples": [
      {
        "name": "Name of sample",
        "uri": "The URI of the sample",
        "extension": "[O] Sample type hint. Values: mpd, ism, m3u8",
        "clip_start_position_ms": "[O] A start point to which the sample should be clipped, in milliseconds"
        "clip_end_position_ms": "[O] An end point from which the sample should be clipped, in milliseconds"
        "drm_scheme": "[O] Drm scheme if protected. Values: widevine, playready, clearkey",
        "drm_license_uri": "[O] URI of the license server if protected",
        "drm_force_default_license_uri": "[O] Whether to force use of "drm_license_uri" for key requests that include their own license URI",
        "drm_key_request_properties": "[O] Key request headers if protected",
        "drm_session_for_clear_content": "[O] Whether to attach a DRM session to clear video and audio tracks"
        "drm_multi_session": "[O] Enables key rotation if protected",
        "subtitle_uri": "[O] The URI of a subtitle sidecar file",
        "subtitle_mime_type": "[O] The MIME type of subtitle_uri (required if subtitle_uri is set)",
        "subtitle_language": "[O] The BCP47 language code of the subtitle file (ignored if subtitle_uri is not set)",
        "ad_tag_uri": "[O] The URI of an ad tag to load via the IMA extension"
      },
      ...etc
    ]
  },
  ...etc
]

Las playlists de Samples se pueden especificar con el siguiente esquema:

[
  {
    "name": "Name of heading",
    "samples": [
      {
        "name": "Name of playlist sample",
        "playlist": [
          {
            "uri": "The URI of the first sample in the playlist",
            "extension": "[O] Sample type hint. Values: mpd, ism, m3u8"
            "clip_start_position_ms": "[O] A start point to which the sample should be clipped, in milliseconds"
            "clip_end_position_ms": "[O] An end point from which the sample should be clipped, in milliseconds"
            "drm_scheme": "[O] Drm scheme if protected. Values: widevine, playready, clearkey",
            "drm_license_uri": "[O] URI of the license server if protected",
            "drm_force_default_license_uri": "[O] Whether to force use of "drm_license_uri" for key requests that include their own license URI",
            "drm_key_request_properties": "[O] Key request headers if protected",
            "drm_session_for_clear_content": "[O] Whether to attach a DRM session to clear video and audio tracks",
            "drm_multi_session": "[O] Enables key rotation if protected",
            "subtitle_uri": "[O] The URI of a subtitle sidecar file",
            "subtitle_mime_type": "[O] The MIME type of subtitle_uri (required if subtitle_uri is set)",
            "subtitle_language": "[O] The BCP47 language code of the subtitle file (ignored if subtitle_uri is not set)"
          },
          {
            "uri": "The URI of the second sample in the playlist",
            ...etc
          },
          ...etc
        ]
      },
      ...etc
    ]
  },
  ...etc
]

Si es necesario, los encabezados de la solicitud de clave se especifican como un objeto que contiene un atributo de string para cada encabezado:

"drm_key_request_properties": {
  "name1": "value1",
  "name2": "value2",
  ...etc
}

En la actividad del selector de muestras, el menú ampliado contiene opciones para especificar si se prefieren los decodificadores de extensiones.

URI de archivos locales y restricciones de almacenamiento específico

Cuando se especifican los URI de archivos locales, la app de demostración solicita los permisos de acceso al almacenamiento necesarios para leer estos archivos. Sin embargo, a partir de Android 13, no es posible cargar archivos arbitrarios que no terminen en una extensión de archivo multimedia típica (como .mp4). Si necesitas cargar un archivo de este tipo, puedes colocarlo en el directorio de almacenamiento específico de la app de demostración que no tenga restricciones de acceso. Por lo general, se encuentra en /sdcard/Android/data/androidx.media3.demo.main/files.

2. Cómo cargar un archivo exolist.json externo

La app de demostración puede cargar archivos JSON externos con el esquema anterior y que tengan un nombre conforme a la convención *.exolist.json. Por ejemplo, si alojas un archivo de este tipo en https://yourdomain.com/samples.exolist.json, puedes abrirlo en la app de demostración de la siguiente manera:

adb shell am start -a android.intent.action.VIEW \
    -d https://yourdomain.com/samples.exolist.json

Si haces clic en un vínculo de *.exolist.json (por ejemplo, en el navegador o en un cliente de correo electrónico) en un dispositivo con la app de demostración instalada, también se abrirá en la app de demostración. Por lo tanto, alojar un archivo JSON *.exolist.json proporciona una forma sencilla de distribuir contenido para que otras personas lo prueben en la app de demostración.

3. Activa un intent

Los intents se pueden usar para omitir la lista de muestras y también iniciarlos directamente en la reproducción. Para reproducir una sola muestra, establece la acción del intent en androidx.media3.demo.main.action.VIEW y su URI de datos en el de la muestra que se reproducirá. Este intent se puede activar desde la terminal mediante lo siguiente:

adb shell am start -a androidx.media3.demo.main.action.VIEW \
    -d https://yourdomain.com/sample.mp4

Los servicios adicionales opcionales admitidos para un solo intent de muestra son los siguientes:

  • Elementos adicionales de configuración de muestra:
    • mime_type [String] Ejemplo de sugerencia de tipo de MIME. Por ejemplo, application/dash+xml para contenido DASH.
    • clip_start_position_ms [Long] Punto de inicio al que se debe recortar la muestra (en milisegundos).
    • clip_end_position_ms [Long] Punto final desde el cual se debe recortar la muestra (en milisegundos).
    • drm_scheme [String] Esquema DRM si está protegido. Los valores válidos son widevine, playready y clearkey. También se aceptan los UUID de esquema DRM.
    • drm_license_uri [String] El URI del servidor de licencias si está protegido.
    • drm_force_default_license_uri [Booleano] Indica si se debe forzar el uso de drm_license_uri para solicitudes de claves que incluyen su propio URI de licencia.
    • drm_key_request_properties [Array de strings] Encabezados de solicitud de clave empaquetados como name1, value1, name2, value2, etc. si están protegidos.
    • drm_session_for_clear_content [Booleano] Indica si se debe conectar una sesión de DRM para borrar pistas de video y audio.
    • drm_multi_session [Booleano] Habilita la rotación de claves si está protegida.
    • subtitle_uri [String] Es el URI de un archivo adicional de subtítulos.
    • subtitle_mime_type [String] Es el tipo de MIME de subtitle_uri (obligatorio si se configura subtitle_uri).
    • subtitle_language [String] El código de idioma BCP47 del archivo de subtítulos (se ignora si no se configura subtitle_uri).
    • ad_tag_uri [String] Es el URI de una etiqueta de anuncio que se cargará con la [extensión de IMA][].
    • prefer_extension_decoders (Booleano) Indica si se prefieren los decodificadores de extensiones en lugar de los de plataforma.

Cuando usas adb shell am start para activar un intent, se puede establecer una string opcional adicional con --es (p.ej., --es extension mpd). Se puede establecer un valor booleano adicional opcional con --ez (p.ej., --ez prefer_extension_decoders TRUE). Se puede configurar un extra largo opcional con --el (p.ej., --el clip_start_position_ms 5000. Se puede configurar un array de cadenas adicional opcional con --esa (p.ej., --esa drm_key_request_properties name1,value1).

Para reproducir una playlist de muestras, establece la acción del intent en androidx.media3.demo.main.action.VIEW_LIST. Los extras de configuración de muestra permanecen iguales que para androidx.media3.demo.main.action.VIEW, excepto por dos diferencias:

  • Las claves de los extras deben tener un guion bajo y el índice basado en 0 de la muestra como sufijo. Por ejemplo, extension_0 sugiere el tipo de muestra para la primera muestra. drm_scheme_1 establecerá el esquema DRM para la segunda muestra.
  • El URI de la muestra se pasa como un extra con la clave uri_<sample-index>.

Otros extras, que no dependen de la muestra, no cambian. Por ejemplo, puedes ejecutar el siguiente comando en la terminal para reproducir una lista de reproducción con dos elementos y anular la extensión del segundo elemento:

adb shell am start -a androidx.media3.demo.main.action.VIEW_LIST \
    --es uri_0 https://a.com/sample1.mp4 \
    --es uri_1 https://b.com/sample2.fake_mpd \
    --es extension_1 mpd