23

从文件路径设置 VideoView 的视频时,我的体验一直不一致。

VideoView myVideoView = findViewById(R.id.videoView);
...
myVideoView.setVideoPath(videoFilename);
...
myVideoView.start();

videoFilename 是我的缓存目录中视频的绝对路径:

String videoFilename = new File(context.getCacheDir(), "myawesomevideo.mp4").getAbsolutePath();

在 Android SDK >= 16 (Jelly Bean) 中,这可以正常工作,并且可以播放我很棒的视频。在 Android 4.0.4 (SDK = 15) 中,调用 myVideoView.start() 时 MediaPlayer 会中断。

错误是永远无益的:

error (1, -2147483648)

我在这里想念什么?直接从我的包资产(res/raw)或互联网( http://something.com/myawesomeinternetvideo.mp4 )加载文件,但我不知道如何从我的缓存目录中读取文件!

4

2 回答 2

15

事实证明,错误 -2147483648 表示未知错误。这可能与视频编码有关,但也值得检查文件路径是否存在以及 VideoView 是否有权读取它

我的问题是我正在使用 Context.MODE_PRIVATE (默认值)编写文件。

openFileOutput(filename, Context.MODE_PRIVATE);

这表明只有您的应用程序可以访问该文件。我不知道具体如何或为什么,但在 Jelly Bean 及更高版本中,似乎允许视频视图访问您指定的文件,就好像它是您的应用程序一样,但在 Jelly Bean 之前,视频视图尝试打开文件在它自己的上下文中(不是你的应用程序的)。由于该模式是私有的,因此它失败了。

一种解决方案是使用Context.MODE_WORLD_READABLE编写文件,现在已弃用。这表明任何人都可以在该路径打开文件。这显然是不安全和不鼓励的。

我最终创建了一个内容提供者和我自己的 URI 来处理这种情况。具体来说:

AndroidManfest.xml

...
    <provider
        android:name="com.myexampleapp.video.VideoProvider"
            android:authorities="com.myexampleapp.video.VideoProvider.files"
        android:exported="false" />
    </application>
</manifest>

视频提供者.java

package com.myexampleapp.video;

import java.io.File;
import java.io.FileNotFoundException;

import android.content.ContentProvider;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.os.ParcelFileDescriptor;

public class VideoProvider extends ContentProvider { 
    public static final Uri CONTENT_URI_BASE =
            Uri.parse("content://com.myexampleapp.video.VideoProvider.files.files/");

    private static final String VIDEO_MIME_TYPE = "video/mp4";

    @Override
    public boolean onCreate() {
        return true;
    }

    @Override
    public String getType(final Uri uri) {
        return VIDEO_MIME_TYPE;
    }

    @Override
    public ParcelFileDescriptor openFile(final Uri uri, final String mode)
            throws FileNotFoundException {
        File f = new File(uri.getPath());

        if (f.exists())
            return (ParcelFileDescriptor.open(f,
                    ParcelFileDescriptor.MODE_READ_ONLY));

        throw new FileNotFoundException(uri.getPath());
    }

    @Override
    public int delete(final Uri uri, final String selection, final String[] selectionArgs) {
        throw new UnsupportedOperationException();
    }

    @Override
    public Uri insert(final Uri uri, final ContentValues values) {
        throw new UnsupportedOperationException();
    }

    @Override
    public Cursor query(final Uri uri, final String[] projection, final String selection, final String[] selectionArgs, final String sortOrder) {
        throw new UnsupportedOperationException();
    }

    @Override
    public int update(final Uri uri, final ContentValues values, final String selection, final String[] selectionArgs) {
        throw new UnsupportedOperationException();
    }
}

然后,我在哪里访问我的视频文件:

VideoView myVideoView = findViewById(R.id.videoView);
...
myVideoView.setVideoURI(
    Uri.parse(
        CachedActionProvider.CONTENT_URI_BASE + Uri.encode(videoFilename)));
...
myVideoView.start();

这是一种非常冗长的方法,它告诉 VideoView 向您的 ContentProvider 询问数据的文件描述符。文件描述符没有权限,因此您使用应用程序的权限打开文件并将其交给 VideoView,而不是要求 VideoView 使用自己的权限打开文件。

这解决了我的问题,希望也解决了你的问题!

于 2013-09-16T17:05:07.947 回答
4

从缓存目录加载视频文件时,我也遇到了这个问题。这个问题似乎只发生在 Nexus 设备上,但我无法在很多设备上测试它。

我尝试使用 Context.MODE_WORLD_READABLE 编写文件,但该修复对我不起作用。VideoProvider.java 也没有修复它。

所以我找到了另一个对我有用的解决方案。我在我的 Activity 中创建了一个 tempFile,并为 VideoView 提供了它的绝对路径。

也许我的代码可以帮助某人:)

 private MediaController mController;

private VideoView mVideoView;

private File mTmpFile = null;

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

    init();
}

private void init() {
    mProgressDialog = new ProgressDialog(this);

    mVideoView = (VideoView) findViewById(R.id.surface_video);
    mVideoView.requestFocus();
    mController = new MediaController(this);
    mVideoView.setMediaController(mController);

    mVideoView.setOnPreparedListener(this);
    mVideoView.setOnCompletionListener(this);
    mVideoView.setOnErrorListener(this);

    String url = getIntent().getExtras().getString(MainActivity.VIDEO_URL_EXTRA_KEY);
    if (VideoLoaderTask.hasVideo(this, url)) {
        url = getCacheDir().toString().concat("/" + VideoLoaderTask.VIDEO_DIR + "/").concat(url);
        copyToTmpFile(url);
        mVideoView.setVideoPath(mTmpFile.getAbsolutePath());
    } else {
        mVideoView.setVideoURI(Uri.parse(url));
    }

    mProgressDialog.show();
    mVideoView.start();
}

@Override
public void onDestroy() {
    if (mTmpFile != null) {
        mTmpFile.delete();
    }
    super.onDestroy();
}

private void copyToTmpFile(String url) {
    File f = new File(url);
    try {
        mTmpFile = File.createTempFile("video", null);
        mTmpFile.deleteOnExit();

        FileInputStream is = new FileInputStream(f);
        int size = is.available();
        byte[] buffer = new byte[size];
        is.read(buffer);
        is.close();

        FileOutputStream fos = new FileOutputStream(mTmpFile);
        fos.write(buffer);
        fos.close();
    } catch (Exception e) {
        mVideoView.setVideoURI(Uri.parse(url));
    }
}
于 2015-03-25T11:07:36.567 回答