Java ExoPlayer-在SD卡中播放本地mp4文件

Java ExoPlayer-在SD卡中播放本地mp4文件,java,android,android-sdcard,exoplayer,Java,Android,Android Sdcard,Exoplayer,我正在使用Exoplayer演示应用程序,希望从SD卡预加载MP4视频。 我已经尝试了从的实现,但它不起作用。在我的exoplayer演示中没有名为DemoUtil.java的类。 取而代之的是: public static final Sample[] LOCAL_VIDEOS = new Sample[] { new Sample("Some User friendly name of video 1", "/mnt/sdcard/video1.mp4", Util.TYPE_OTHER),

我正在使用Exoplayer演示应用程序,希望从SD卡预加载MP4视频。 我已经尝试了从的实现,但它不起作用。在我的exoplayer演示中没有名为DemoUtil.java的类。 取而代之的是:

public static final Sample[] LOCAL_VIDEOS = new Sample[] {
new Sample("Some User friendly name of video 1",
"/mnt/sdcard/video1.mp4", Util.TYPE_OTHER),
};
我也不能使用SampleChooseActivity.java中提到的他们的代码片段。(不断给我错误)

我改为:

group = new SampleGroup("Local Videos");
group.addAll(Samples.LOCAL_VIDEOS);
sampleGroups.add(group);

我做错了什么?是否每个设备的文件路径都会更改?

在某些设备中,您可以直接使用此路径“/sdcard/nameoffile.mp4”。

使用以下代码从sd卡播放视频。我的测试文件位于sd卡的视频目录中

public static final Sample[] LOCAL_VIDEOS = new Sample[] {
        new Sample("test",
            Environment.getExternalStorageDirectory()+"/Videos/test.mp4", Util.TYPE_OTHER),
};

尚未尝试演示应用程序,但我已成功创建了自己播放本地音频文件的示例,并将其发布在此处:

以下是完成从文件Uri准备播放器的所有工作的主要部分:

private void prepareExoPlayerFromFileUri(Uri uri){
        exoPlayer = ExoPlayerFactory.newSimpleInstance(this, new DefaultTrackSelector(null), new DefaultLoadControl());
        exoPlayer.addListener(eventListener);

        DataSpec dataSpec = new DataSpec(uri);
        final FileDataSource fileDataSource = new FileDataSource();
        try {
            fileDataSource.open(dataSpec);
        } catch (FileDataSource.FileDataSourceException e) {
            e.printStackTrace();
        }

        DataSource.Factory factory = new DataSource.Factory() {
            @Override
            public DataSource createDataSource() {
                return fileDataSource;
            }
        };
        MediaSource audioSource = new ExtractorMediaSource(fileDataSource.getUri(),
                factory, new DefaultExtractorsFactory(), null, null);

        exoPlayer.prepare(audioSource);
    }
您可以像这样获取Uri:
Uri.fromFile(file)

如上所示准备好要播放的文件后,只需调用
exoPlayer.setplaywhenrady(true);
即可开始播放


对于视频文件,您可能只需要将曲面视图附加到exoPlayer对象,但我还没有用ExoPlayer2真正做到这一点。

对于那些希望使用exoPlayer 2从资源播放视频的人,这里有一种方法:

String playerInfo = Util.getUserAgent(context, "ExoPlayerInfo");
DefaultDataSourceFactory dataSourceFactory = new DefaultDataSourceFactory(
        context, playerInfo
);
MediaSource mediaSource = new ExtractorMediaSource.Factory(dataSourceFactory)
    .setExtractorsFactory(new DefaultExtractorsFactory())
    .createMediaSource(Uri.parse("asset:///your_video.mov"));
player.prepare(mediaSource);

这对我很有效。请尝试以下步骤:

获取文件路径并启动播放器

File myFile = new File(extStore.getAbsolutePath() + "/folder/videos/" + video_name);  
videoUrl= String.valueOf(Uri.fromFile(myFile));  
initializePlayer(videoUrl);
初始化播放器

private void initializePlayer(String videoUrl) {
    player = ExoPlayerFactory.newSimpleInstance(
            new DefaultRenderersFactory(getActivity()),
            new DefaultTrackSelector(), new DefaultLoadControl());

    playerView.setPlayer(player);

    player.setPlayWhenReady(playWhenReady);
    player.seekTo(currentWindow, playbackPosition);

    Uri uri = Uri.parse(videoUrl);
    MediaSource mediaSource = buildMediaSource(uri);
    player.prepare(mediaSource, resetPositionBoolean, false);
}   
建筑媒体资源

  private MediaSource buildMediaSource(Uri uri) {
    return new ExtractorMediaSource.Factory(
            new DefaultDataSourceFactory(getActivity(),"Exoplayer-local")).
            createMediaSource(uri);
}

对于希望从资产文件夹加载的用户:

const val VIDEO_URL = "asset:///sample.mkv"
资产/xyz.mp4

它通过以下方式加载文件:

“文件:/android_asset/xyz.mp4”


使用DefaultDataSourceFactory。在exoplayer 2.4.4上检查。

对于那些试图从res/raw*文件夹播放视频文件的人,以下是解决方案。请记住,我使用了**2.8.0版本的exoplayer

public class MainActivity extends AppCompatActivity {

PlayerView playerView;
SimpleExoPlayer simpleExoPlayer;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    playerView=findViewById(R.id.playerView);
}

@Override
protected void onStart() {
    simpleExoPlayer= ExoPlayerFactory.newSimpleInstance(this,new DefaultTrackSelector());
    DefaultDataSourceFactory defaultDataSourceFactory=new DefaultDataSourceFactory(this, Util.getUserAgent(this,"YourApplicationName"));
    simpleExoPlayer.setPlayWhenReady(true);
    ExtractorMediaSource extractorMediaSource=new ExtractorMediaSource.Factory(defaultDataSourceFactory).createMediaSource(RawResourceDataSource.buildRawResourceUri(R.raw.video));
    simpleExoPlayer.prepare(extractorMediaSource);
    playerView.setPlayer(simpleExoPlayer);

    super.onStart();
}

@Override
protected void onStop() {
    playerView.setPlayer(null);
    simpleExoPlayer.release();
    simpleExoPlayer=null;
    super.onStop();
}

}

如果您需要从
资产
字段获取数据,这将起作用。这不适用于从SD卡获取的数据

package com.studio.mattiaferigutti.exoplayertest

import android.annotation.SuppressLint
import android.net.Uri
import android.os.Bundle
import android.util.Log
import android.view.View
import androidx.appcompat.app.AppCompatActivity
import com.google.android.exoplayer2.*
import com.google.android.exoplayer2.source.ProgressiveMediaSource
import com.google.android.exoplayer2.trackselection.DefaultTrackSelector
import com.google.android.exoplayer2.upstream.*
import com.google.android.exoplayer2.upstream.DataSource.Factory
import com.google.android.exoplayer2.util.Util
import kotlinx.android.synthetic.main.activity_main.*


class MainActivity : AppCompatActivity(), Player.EventListener {

    private var player: SimpleExoPlayer? = null
    private var playWhenReady = true
    private var currentWindow = 0
    private var playbackPosition: Long = 0

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
    }

    public override fun onStart() {
        super.onStart()
        if (Util.SDK_INT > 23) {
            initializePlayer("assets:///your_file.your_extension")
        }
    }

    public override fun onResume() {
        super.onResume()
        hideSystemUi()
        if (Util.SDK_INT <= 23 || player == null) {
            initializePlayer("assets:///your_file.your_extension")
        }
    }

    public override fun onPause() {
        super.onPause()
        if (Util.SDK_INT <= 23) {
            releasePlayer()
        }
    }

    public override fun onStop() {
        super.onStop()
        if (Util.SDK_INT > 23) {
            releasePlayer()
        }
    }

    private fun initializePlayer(path: String) {
        if (player == null) {
            val trackSelector = DefaultTrackSelector(this)
            trackSelector.setParameters(
                trackSelector.buildUponParameters().setMaxVideoSizeSd())
            player = SimpleExoPlayer.Builder(this)
                .setTrackSelector(trackSelector)
                .build()
        }
        video_view?.player = player
        video_view?.requestFocus()

        val dataSourceFactory = Factory { AssetDataSource(this@MainActivity) }

        val videoSource = ProgressiveMediaSource.Factory(dataSourceFactory)
            .createMediaSource(Uri.parse(path))

        player?.playWhenReady = playWhenReady
        player?.seekTo(currentWindow, playbackPosition)
        player?.addListener(this)
        player?.prepare(videoSource)
    }

    private fun releasePlayer() {
        if (player != null) {
            playbackPosition = player?.currentPosition!!
            currentWindow = player?.currentWindowIndex!!
            playWhenReady = player?.playWhenReady!!
            player?.removeListener(this)
            player?.release()
            player = null
        }
    }

    /**
     * set fullscreen
     */
    @SuppressLint("InlinedApi")
    private fun hideSystemUi() {
        video_view?.systemUiVisibility = (View.SYSTEM_UI_FLAG_LOW_PROFILE
                or View.SYSTEM_UI_FLAG_FULLSCREEN
                or View.SYSTEM_UI_FLAG_LAYOUT_STABLE
                or View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY
                or View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
                or View.SYSTEM_UI_FLAG_HIDE_NAVIGATION)
    }

    override fun onPlayerError(error: ExoPlaybackException) {
        super.onPlayerError(error)

        //handle the error
    }

    override fun onPlayerStateChanged(playWhenReady: Boolean, playbackState: Int) {
        val stateString: String = when (playbackState) {
            ExoPlayer.STATE_IDLE -> "ExoPlayer.STATE_IDLE      -"
            ExoPlayer.STATE_BUFFERING -> "ExoPlayer.STATE_BUFFERING -"
            ExoPlayer.STATE_READY -> "ExoPlayer.STATE_READY     -"
            ExoPlayer.STATE_ENDED -> "ExoPlayer.STATE_ENDED     -"
            else -> "UNKNOWN_STATE             -"
        }
        Log.d(TAG, "changed state to " + stateString
                + " playWhenReady: " + playWhenReady)
    }

    companion object {
        private val TAG = MainActivity::class.java.name
    }
}
package com.studio.mattiaferigutti.exoplayertest
导入android.annotation.SuppressLint
导入android.net.Uri
导入android.os.Bundle
导入android.util.Log
导入android.view.view
导入androidx.appcompat.app.appcompat活动
导入com.google.android.exoplayer2*
导入com.google.android.exoplayer2.source.ProgressiveMediaSource
导入com.google.android.exoplayer2.trackselection.DefaultTrackSelector
导入com.google.android.exoplayer2.upstream*
导入com.google.android.exoplayer2.upstream.DataSource.Factory
导入com.google.android.exoplayer2.util.util
导入kotlinx.android.synthetic.main.activity\u main*
类MainActivity:AppCompatActivity(),Player.EventListener{
私有变量播放器:SimpleExoPlayer?=null
private var playwhenrady=true
私有变量currentWindow=0
专用变量播放位置:长=0
重写创建时的乐趣(savedInstanceState:Bundle?){
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
}
public override fun onStart(){
super.onStart()
如果(Util.SDK_INT>23){
初始化图层(“assets:///your_file.your_extension")
}
}
恢复时的公共覆盖功能(){
super.onResume()
hideSystemUi()
如果(Util.SDK_INT“ExoPlayer.STATE_IDLE”——
ExoPlayer.STATE_缓冲->“ExoPlayer.STATE_缓冲--”
ExoPlayer.STATE_就绪->“ExoPlayer.STATE_就绪--”
ExoPlayer.STATE_END->“ExoPlayer.STATE_END——”
else->“未知状态”
}
Log.d(标记,“已将状态更改为”+stateString
+“playWhenReady:”+playWhenReady)
}
伴星{
private val TAG=MainActivity::class.java.name
}
}

对于Exo player 2.13.3

implementation 'com.google.android.exoplayer:exoplayer:2.13.3'
将视频放入“资源”文件夹:

const val VIDEO_URL = "asset:///sample.mkv"
初始化播放器:

val mediaItem: MediaItem = MediaItem.fromUri(VIDEO_URL)
player.setMediaItem(mediaItem)

完美答案…毫无疑问:)所有这些代码只是为了将文件路径传递给ExoPlayer,这只需一行就可以完成视频查看?天哪,ExoPlayer的设计者一定是疯子或虐待狂。正如所说,
extractoremiasource
不推荐使用(SDK 29)。我知道。当我看到这个解决方案时,我感到愤怒和不安。而且,我很沮丧没有找到一个简单的单行API来播放媒体文件,就像我们使用
MediaPlayer
一样。我想我可能只是在寻找一个替代方案,uggh。现在这太复杂了:/