Android - загрузка видео из личной папки приложения - PullRequest
4 голосов
/ 30 января 2012

У меня есть проблема, которую я долго пытался найти решение.Моя ситуация выглядит следующим образом:

У меня есть приложение, которое загружает сжатые видеоролики и распаковывает их в личной папке приложения, а точнее в подпапке.Например, в / data / data / my.app.package.name.here / files / assets / assets-955.

Внутри этой папки видео разархивировано.Процесс распаковки успешно завершен, так как я могу без проблем извлекать и просматривать видео при запуске приложения на эмуляторе.

Затем у меня появляется другая операция, которая обращается к этой папке, находит видеофайл и пытается открыть его.,В этот момент я получаю сообщение об ошибке «Извините, это видео не может быть воспроизведено» со следующим стеком ошибок:

01-30 17:36:17.770: D/ContentDemoActivity(6757): File: /data/data/xxxx/files/assets/assets-955/bank_2.mp4
01-30 17:36:17.830: I/MediaPlayer(6757): prepareAsync called in state 4
01-30 17:36:17.830: E/MediaPlayer(6757): error (1, -2147483648)
01-30 17:36:17.860: E/MediaPlayer(6757): Error (1,-2147483648)
01-30 17:36:17.860: D/VideoView(6757): Error: 1,-2147483648
01-30 17:36:19.370: E/MediaPlayer(6757): stop called in state 0
01-30 17:36:19.370: E/MediaPlayer(6757): error (-38, 0)
01-30 17:36:19.370: W/MediaPlayer(6757): mediaplayer went away with unhandled events

Код, с которым я пытаюсь воспроизвести видео, довольно прост:

mView = (VideoView) findViewById(R.id.videoView);

mMediaPlayer = new MediaPlayer();

mView.requestFocus();
mHolder = mView.getHolder();

Log.d(tag, "Populating content. Assets path: " + mAssetsPath);
File f = new File(mAssetsPath);
File[] files = f.listFiles();
Log.d(tag, "File: " + files[0].toString());

mView.setVideoURI(Uri.parse(files[0].toString()));

mView.setMediaController(new MediaController(this));

и макет действия имеет простой VideoView, ничего особенного там нет.

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

Uri video = Uri.parse("android.resource://" + getPackageName() + "/" + R.raw.bank_2);
mVideoView.setVideoURI(video);
mVideoView.start();

Я бы сделал то же самое с загруженными видео, но в API нет функции, которая позволила бы мне загружать видеоUri из личной папки приложения.

Я нашел различные решения, используя файловые дескрипторы, прослушиватели для videoView, флаги, указывающие MODE_WORLD_READABLE, предварительный расчет размеров videoView и т. Д., Но ни один из них не дал положительных результатов..

В двух словах, мои вопросы:

  • Почему я получаю те ошибки, которые в соответствии с тем, что я нашел в Интернете, являются ошибками, которые связаны спроблемное кодирование видео файла?
  • Что лучше всего использовать в моем случае, VideoView или SurfaceView?
  • Какой идеальный метод для загрузки видео из личной папки приложения и возможности его воспроизведения?

Спасибо.

РЕДАКТИРОВАТЬ

После предложения CommonsWare я пошел со следующей реализацией:

File f = new File(mAssetsPath);
File[] files = f.listFiles();
Log.d(tag, "File: " + files[0].toString());

URI uri = URI.create("file://" + (files[0].toString()));
File file = new File(uri);
try {
    Log.d(tag, "1");
    ParcelFileDescriptor parcel = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_WRITE);
   Log.d(tag, "2");
    mMediaPlayer.setDataSource(parcel.getFileDescriptor());
    Log.d(tag, "3");
    mMediaPlayer.start();
    Log.d(tag, "4");
} catch (FileNotFoundException e) {
    e.printStackTrace();
} catch (IllegalArgumentException e) {
    e.printStackTrace();
} catch (IllegalStateException e) {
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}

 Log.d(tag, "5");

К сожалению, на этот раз я получаюследующие ошибки:

01-31 12:40:11.480: D/ContentDemoActivity(15896): File: /data/data/com.houseofradon.meb/files/assets/assets-955/bank_2.mp4
01-31 12:40:11.480: D/ContentDemoActivity(15896): 1
01-31 12:40:11.480: D/ContentDemoActivity(15896): 2
01-31 12:40:11.500: D/ContentDemoActivity(15896): 3
01-31 12:40:11.500: E/MediaPlayer(15896): start called in state 2
01-31 12:40:11.500: E/MediaPlayer(15896): error (-38, 0)
01-31 12:40:11.500: D/ContentDemoActivity(15896): 4
01-31 12:40:11.500: D/ContentDemoActivity(15896): 5
01-31 12:40:11.530: E/MediaPlayer(15896): Error (-38,0)

Итак, что-то происходит при запуске медиаплеера.Код ошибки -38, кажется, не означает ничего конкретного, как я обнаружил здесь .

Есть идеи, что мне не хватает ???

EDIT # 2

Теперь я использую mediaPlayer и SurfaceView для выполнения всего процесса вместе со слушателем SurfaceHolder.Вот код:

mMediaPlayer = new MediaPlayer();

mSurfaceView = (SurfaceView) findViewById(R.id.surface);
mHolder = mSurfaceView.getHolder();
mHolder.addCallback(this);


public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    Log.d(tag, "surfaceChanged");

    try {
        mMediaPlayer.setDisplay(mHolder);
        Log.d(tag, "7");
        mMediaPlayer.start();
        Log.d(tag, "8");
    } catch (IllegalStateException e) {
        e.printStackTrace();
    }

    Log.d(tag, "9");
}



public void surfaceCreated(SurfaceHolder holder) {
    Log.d(tag, "surfaceCreated");
    File f = new File(mAssetsPath);
    File[] files = f.listFiles();
    Log.d(tag, "File: " + files[0].toString());

    URI uri = URI.create("file://" + (files[0].toString()));
    File file = new File(uri);
    try {
        Log.d(tag, "1");
        ParcelFileDescriptor parcel = ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_WRITE);
        Log.d(tag, "2");
        mMediaPlayer.setDataSource(parcel.getFileDescriptor());
        Log.d(tag, "3");
        mMediaPlayer.setVolume(100, 100);
        Log.d(tag, "4");
        mMediaPlayer.prepare();
        Log.d(tag, "5");

    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IllegalArgumentException e) {
        e.printStackTrace();
    } catch (IllegalStateException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }

    Log.d(tag, "6");
}

Я могу слушать аудио видео, но картинка просто черного цвета.Я также получаю сообщение об ошибке почти в конце воспроизведения видео, которое говорит:

01-31 14:26:01.300: W/AudioSystem(17165): AudioFlinger server died!
01-31 14:26:01.300: W/IMediaDeathNotifier(17165): media server died
01-31 14:26:01.300: E/MediaPlayer(17165): error (100, 0)
01-31 14:26:01.300: E/MediaPlayer(17165): Error (100,0)

Я использую реальное устройство, Samsung Galaxy Tab 10.1.Есть идеи?

1 Ответ

4 голосов
/ 30 января 2012

Почему я получаю те ошибки, которые в соответствии с тем, что я обнаружил в сети, являются ошибками, связанными с проблемным кодированием видеофайла?

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

Что лучше всего использовать в моем случае, VideoView или SurfaceView?

AVideoView содержит SurfaceView.Используете ли вы VideoView или комбинацию MediaPlayer и SurfaceView.

Какой идеальный способ загрузить видео из личной папки приложения и иметь возможность его воспроизвести?

Либо создайте ContentProvider, который может обслуживать ваш локальный файл, и используйте поставщика Uri вместо Uri для локального файла, либо создайте локальный файл с помощью openFileOutput() иMODE_WORLD_READABLE.

...