Exoplayer адаптивная потоковая передача hls

Я ищу хороший и простой пример/объяснение, как реализовать ExoPlayer для потоковой передачи HLS Adaptive. Я новичок и у меня нет опыта и знаний, поэтому я могу понять, как это сделать, из примера кода на git.

Слишком много «движущихся частей», поэтому новичок может понять и использовать их в своих проектах.

Может ли кто-нибудь помочь мне узнать и понять, как использовать/реализовать ExoPlayer для достижения этой функциональности?

Спасибо!


person svarog    schedule 14.02.2017    source источник


Ответы (2)


Самый простой способ начать использовать ExoPlayer — добавить его в качестве зависимости gradle. Вам необходимо убедиться, что репозиторий jcenter включен в файл build.gradle в корне вашего проекта:

repositories { jcenter() }

Затем включите в файл build.gradle вашего модуля следующее:

compile 'com.google.android.exoplayer:exoplayer:r2.2.0'

<сильный>1. Ваш файл макета

    <?xml version="1.0" encoding="utf-8"?>
<FrameLayout
    android:id="@+id/activity_main"
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.google.android.exoplayer2.ui.SimpleExoPlayerView
        android:id="@+id/player_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:focusable="true"
        app:resize_mode="fill"/>

    <ProgressBar
        android:id="@+id/progressBar"
        style="?android:attr/progressBarStyle"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:visibility="gone"/>

</FrameLayout>

<сильный>2. Ваш файл класса (активность)

    public class VideoPlayerActivity extends AppCompatActivity implements ExoPlayer.EventListener {

    private SimpleExoPlayerView simpleExoPlayerView;
    private String hlsVideoUri = "http://playertest.longtailvideo.com/adaptive/bbbfull/bbbfull.m3u8";
    private SimpleExoPlayer player;
    private ProgressBar progressBar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_video_player);

        // 1. Create a default TrackSelector
        Handler mainHandler = new Handler();
        BandwidthMeter bandwidthMeter = new DefaultBandwidthMeter();
        TrackSelection.Factory videoTrackSelectionFactory = new AdaptiveVideoTrackSelection.Factory(bandwidthMeter);
        TrackSelector trackSelector = new DefaultTrackSelector(videoTrackSelectionFactory);

        // 2. Create a default LoadControl
        LoadControl loadControl = new DefaultLoadControl();


        // 3. Create the player
        player = ExoPlayerFactory.newSimpleInstance(this, trackSelector, loadControl);

        simpleExoPlayerView = (SimpleExoPlayerView) findViewById(R.id.player_view);
        simpleExoPlayerView.setPlayer(player);

        // Measures bandwidth during playback. Can be null if not required.
        DefaultBandwidthMeter defaultBandwidthMeter = new DefaultBandwidthMeter();
        // Produces DataSource instances through which media data is loaded.
        DataSource.Factory dataSourceFactory = new DefaultDataSourceFactory(this,
                Util.getUserAgent(this, "Exo2"), defaultBandwidthMeter);
        // Produces Extractor instances for parsing the media data.
        ExtractorsFactory extractorsFactory = new DefaultExtractorsFactory();
        // This is the MediaSource representing the media to be played.
        HlsMediaSource hlsMediaSource = new HlsMediaSource(Uri.parse(hlsVideoUri), dataSourceFactory, mainHandler, new AdaptiveMediaSourceEventListener() {
            @Override
            public void onLoadStarted(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs) {

            }

            @Override
            public void onLoadCompleted(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs, long loadDurationMs, long bytesLoaded) {

            }

            @Override
            public void onLoadCanceled(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs, long loadDurationMs, long bytesLoaded) {

            }

            @Override
            public void onLoadError(DataSpec dataSpec, int dataType, int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaStartTimeMs, long mediaEndTimeMs, long elapsedRealtimeMs, long loadDurationMs, long bytesLoaded, IOException error, boolean wasCanceled) {

            }

            @Override
            public void onUpstreamDiscarded(int trackType, long mediaStartTimeMs, long mediaEndTimeMs) {

            }

            @Override
            public void onDownstreamFormatChanged(int trackType, Format trackFormat, int trackSelectionReason, Object trackSelectionData, long mediaTimeMs) {

            }
        });

        player.addListener(this);
        player.prepare(hlsMediaSource);
        simpleExoPlayerView.requestFocus();
        player.setPlayWhenReady(true);

        progressBar = (ProgressBar) findViewById(R.id.progressBar);
    }

    @Override
    public void onTimelineChanged(Timeline timeline, Object manifest) {

    }

    @Override
    public void onTracksChanged(TrackGroupArray trackGroups, TrackSelectionArray trackSelections) {

    }

    @Override
    public void onLoadingChanged(boolean isLoading) {

    }

    @Override
    public void onPlayerStateChanged(boolean playWhenReady, int playbackState) {

        switch (playbackState) {
            case Player.STATE_BUFFERING:
                //You can use progress dialog to show user that video is preparing or buffering so please wait
                progressBar.setVisibility(View.VISIBLE);
                break;
            case Player.STATE_IDLE:
                //idle state
                break;
            case Player.STATE_READY:
                // dismiss your dialog here because our video is ready to play now
                progressBar.setVisibility(View.GONE);
                break;
            case Player.STATE_ENDED:
                // do your processing after ending of video
                break;
        }
    }

    @Override
    public void onPlayerError(ExoPlaybackException error) {

        AlertDialog.Builder adb = new AlertDialog.Builder(VideoPlayerActivity.this);
        adb.setTitle("Could not able to stream video");
        adb.setMessage("It seems that something is going wrong.\nPlease try again.");
        adb.setPositiveButton("OK", new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                dialog.dismiss();
                finish(); // take out user from this activity. you can skip this
            }
        });
        AlertDialog ad = adb.create();
        ad.show();
    }

    @Override
    public void onPositionDiscontinuity() {

    }

    @Override
    protected void onPause() {
        super.onPause();
        if (player != null) {
            player.setPlayWhenReady(false); //to pause a video because now our video player is not in focus
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        player.release();
    }
}

Думаю для новичка этого достаточно. Также имейте в виду, что стандартные аудио- и видеокомпоненты этой библиотеки зависят от Android MediaCodec API, который был выпущен в Android 4.1 (уровень API 16). Так что на андроиде 4.0 и ниже работать не будет.

Не забудьте добавить это разрешение в manifest file :

<uses-permission android:name="android.permission.INTERNET"/>

person Vicky    schedule 21.02.2017
comment
Спасибо Вики! Скажите, будет ли exoplayer автоматически выбирать качество видео в зависимости от пропускной способности? Я вижу методы, которые должны быть реализованы, но я не понимаю, как я могу это использовать? Пожалуйста, обнажись со мной. Спасибо :) - person svarog; 21.02.2017
comment
Exoplayer автоматически выбирает видео по пропускной способности. Вам не нужно беспокоиться об этом. - person Vicky; 21.02.2017
comment
Вы получили BehindLiveWindowException за этот код? Я вижу ошибку, связанную с этой проблемой, на github. У меня есть метод переопределения onPlayerError из интерфейса ExoPlayer.EventListener, но не повезло. Какие-либо предложения? - person svarog; 22.02.2017
comment
конечно, вот класс viewModel, в котором я реализовал плеер - person svarog; 22.02.2017
comment
Думаю, это баг, на гитхабе говорят переинициализировать медиаисточник, вы это уже сделали. - person Vicky; 22.02.2017
comment
@ Вики, предположим, у меня есть 720p видео на сервере, и я хочу, чтобы оно воспроизводилось в соответствии с пропускной способностью сети. Exoplaer делает это автоматически? - person Sagar; 22.02.2018
comment
@Vikcy Есть ли способ адаптивной потоковой передачи RTMP с помощью exo player? - person Nitish; 20.09.2019

Ответ от @Vicky будет работать, но имеет один недостаток.

Измеритель пропускной способности, который вы передаете селектору дорожек, должен быть тем же, что и фабрика источников данных. Фабрика источников данных поддерживает оценку пропускной способности, вызывая методы измерителя BW, а процесс выбора адаптивной дорожки получает оценку, чтобы решить, к какой дорожке адаптироваться.

Если они не являются одним и тем же экземпляром, адаптивный выбор всегда получает -1 в качестве BW и выбирает какой-то средний вариант.

Демонстрационные приложения ExoPlayer также имеют этот недостаток. Они передают false в useBwMeter в buildDataSource(), что означает отсутствие обновления оценки BW. Нет необходимости использовать измеритель BW.

person Darren    schedule 14.03.2018
comment
Есть ли способ адаптивной потоковой передачи RTMP с помощью exo player? - person Nitish; 20.09.2019