У меня есть проблема, которую я долго пытался найти решение.Моя ситуация выглядит следующим образом:
У меня есть приложение, которое загружает сжатые видеоролики и распаковывает их в личной папке приложения, а точнее в подпапке.Например, в / 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.Есть идеи?