Как пропустить последний кадр при задержке видео при воспроизведении потока UDP по LibVL C in Android Kotlin - PullRequest
0 голосов
/ 02 мая 2020

Я пытаюсь воспроизвести поток UDP с помощью LibVL C в Android Box, но у меня есть некоторые проблемы. На некоторых каналах HD видео отстает и останавливается, но звук в порядке и продолжается. Я перепробовал множество опций, но это не сработало.

это мой класс контроллеров Vl c:

class VideoController(activity: AppCompatActivity, context: Context) : IVLCVout.Callback ,
    MediaPlayer.EventListener {

// create TAG for logging
companion object {
    var TAG = "VideoController"
}

// declare media player object
var mediaPlayer: MediaPlayer? = null
// declare surface view object
var mSurface: SurfaceView? = null
// declare surface holder object
var holder: SurfaceHolder? = null

// declare libvlc object
private var libvlc: LibVLC? = null

// declare/initialize activity
var activity: Activity? = null
var context: Context? = null

init {
    this.activity = activity
    this.context = context
}

/////////////////////


/////////////////////
fun createPlayer(media: String) {

    if (mediaPlayer != null && libvlc != null) {
        releasePlayer()
    }
    Log.i(TAG, "Creating vlc player")

    try {
        // create arraylist to assign option to create libvlc object
        val options = ArrayList<String>()
        options.add("--audio-time-stretch"); // time stretching
        options.add("-vvv") // verbosity
        options.add("--no-sub-autodetect-file")
        options.add("--swscale-mode=0")
        options.add("--network-caching=5000")
        options.add("--drop-late-frames")
        options.add("--skip-frames")
        options.add("--avcodec-skip-frame")
        options.add("--avcodec-hw=any")
        options.add("--aout=opensles")
        options.add("--rtsp-tcp")
        options.add("--http-reconnect")
        options.add("--sout-display")
        options.add("--sout=any")
        options.add("--packetizer=any")
        options.add("--packetizer-mpegvideo-sync-iframe")
        options.add("--avcodec-skip-frame=4")
        options.add("--network-synchronisation")
        options.add("--sout-x264-lookahead=60")
        options.add("--avcodec-hurry-up")
        options.add("--audio-desync=1")
        options.add("--no-hdtv-fix")
        options.add("--udp-buffer=5000")
        options.add("--sout-rtp-proto=udp")


        // create libvlc object
        libvlc = LibVLC(activity, options)

        // get surface view holder to display video
        this.holder = mSurface!!.holder
        holder!!.setKeepScreenOn(true)

        // Creating media player
        mediaPlayer = MediaPlayer(libvlc)

        // Setting up video output
        val vout = mediaPlayer!!.vlcVout
        vout.setVideoView(mSurface)
        vout.addCallback(this)
        vout.attachViews()
        val m = Media(libvlc, Uri.parse(media))
        m.setHWDecoderEnabled(true, false)
        m.addOption(":network-caching=5000")
        m.addOption(":clock-jitter=1")
        m.addOption(":clock-synchro=1")
        m.addOption(":access=udp")
        m.addOption(":drop-late-frames")
//            m.addOption(":codec=all")  //---  I tried this option too, but it lagging at all

        mediaPlayer!!.isSeekable
        mediaPlayer!!.setMedia(m)
        mediaPlayer!!.play()


        mediaPlayer!!.setEventListener { event: MediaPlayer.Event? ->
            when(event?.type) {
                MediaPlayer.Event.Opening -> {
                    Log.i(TAG, "Event Opening")
                }
                MediaPlayer.Event.Buffering -> {
                    Log.i(TAG, "Event Buffering=" + event.getBuffering());
                }
                MediaPlayer.Event.Stopped -> {
                Log.i(TAG , "Event Stopped")
                }

                MediaPlayer.Event.EncounteredError -> {
                    Log.i(TAG , "Hadi paused")

                }
            }
        }


        val cm =
            context?.getSystemService(Context.CONNECTIVITY_SERVICE) as ConnectivityManager
        val progressThread = Thread(
            Runnable {
                kotlin.run {
                    try {
                        while (mediaPlayer!=null && !mediaPlayer!!.isPlaying()){}

                        var temp = 0
                        while (mediaPlayer!!.isPlaying()) {

                            val activeNetwork: NetworkInfo? = cm.activeNetworkInfo
                            val isConnected: Boolean? = activeNetwork?.isConnectedOrConnecting
                            if (isConnected == false ) {
                                activity?.runOnUiThread {                                        
                                }
                                temp = 1
                            } else if (isConnected==true && temp == 1) {
                                activity?.runOnUiThread {
                                }
                                //releasePlayer()
                                mSurface!!.requestFocus()
                                mSurface!!.refreshDrawableState()
                                mediaPlayer!!.setMedia(m)
                                mediaPlayer!!.play()
                                temp=0
                                while (!mediaPlayer!!.isPlaying()){}

                            }

                        }
                    } catch (e: java.lang.Exception) {

                    }
                }
            })

        progressThread.start()

    } catch (e: Exception) {
        Toast.makeText(
            activity, "Error in creating player!", Toast
                .LENGTH_LONG
        ).show()
    }

}

fun getWifiLevel(): Int {
    val wifiManager =
        context!!.getSystemService(Context.WIFI_SERVICE) as WifiManager
    val linkSpeed = wifiManager.connectionInfo.rssi
    return WifiManager.calculateSignalLevel(linkSpeed, 5)
}


fun stopAll() {
    mediaPlayer?.stop()
    mediaPlayer = null
    holder = null
    libvlc = null
}

/*
* release player
* */
fun releasePlayer() {
    Log.i(TAG, "releasing player started")
    if (libvlc == null)
        return
    mediaPlayer!!.stop()
    var vout: IVLCVout = mediaPlayer!!.vlcVout
    vout.removeCallback(this)
    vout.detachViews()
    mediaPlayer!!.release()
    mediaPlayer = null
    holder = null
    libvlc!!.release()
    libvlc = null

    Log.i(TAG, "released player")
}

override fun onEvent(event: MediaPlayer.Event) {

    when (event.type) {
        MediaPlayer.Event.EndReached -> {
            this.releasePlayer()
        }

        MediaPlayer.Event.EncounteredError -> {
            Log.i("vlc-lib", "Error Play Video")
            mediaPlayer?.play()
        }

        MediaPlayer.Event.Playing -> Log.i("playing", "playing")
        MediaPlayer.Event.Paused -> {
            Log.i("paused", "paused")
            mediaPlayer?.play()
        }
        MediaPlayer.Event.Stopped -> {
            Log.i("stopped", "stopped")
            mediaPlayer?.play()
        }
        else -> Log.i("nothing", "nothing")
    }
}

override fun onSurfacesCreated(vlcVout: IVLCVout?) {
    val sw = mSurface!!.width
    val sh = mSurface!!.height

    if (sw * sh == 0) {
        Log.e(TAG, "Invalid surface size")
        return
    }

    mediaPlayer!!.vlcVout.setWindowSize(sw, sh)

    mediaPlayer!!.aspectRatio = "Wide-Screen"
    mediaPlayer!!.setScale(0f)
}

override fun onSurfacesDestroyed(vlcVout: IVLCVout?) {
    releasePlayer()
}

}

В любое время, когда я хочу играть, я использую его createPlayer:

videoController!!.createPlayer("udp://@239.255.2.1:1234")

И это моя активность:

    <?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

<SurfaceView
    android:id="@+id/video_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center"
    android:keepScreenOn="true"
    app:layout_constraintTop_toTopOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintBottom_toBottomOf="parent"
    />

</androidx.constraintlayout.widget.ConstraintLayout>

Я перепробовал множество опций, но это не сработало, и я добавил --skip-late-frames к опциям.

Спасибо за помощь

...