現在、ビデオを内部ストレージにダウンロードして(残念ながらSDカードを持っていません)、単に再生するアプリケーションに取り組んでいます。私がこれまでに試したことは次のとおりです。
標準の VideoView、/data/data/.../file.mp4 をパスとして設定 - 機能しませんでした。SurfaceView で MediaPlayer を使用し、パスまたはファイル記述子を使用 - 機能しませんでした。
私が今ここに持っているのは、VideoView のわずかに変更されたバージョンで、次の変更があります。
if(mUri != null)
{
mMediaPlayer.setDataSource(mContext, mUri);
}
else
{
mMediaPlayer.setDataSource(mFd);
}
また、ファイル記述子を設定できる setFD メソッドもあります。これは、MediaPlayer/SurfaceView 全体をアクティビティに入れるよりもはるかに簡単でクリーンです。問題は次のとおりです。
内部ストレージのファイルは次のように作成されます。
FileInputStream fis = openFileInput(Downloader.TMP_FILENAME);
FileOutputStream fos = openFileOutput(Downloader.FILENAME
+ ext, Context.MODE_WORLD_READABLE);
byte[] buf = new byte[2048];
while (fis.read(buf) > 0) {
fos.write(buf);
}
fis.close();
fos.close();
deleteFile(Downloader.TMP_FILENAME);
Log.d(TAG, "Replacement done!");
基本的には、一時ファイルを読み取り(現在再生されているファイルを上書きしたくないため..しかし、それは実際には問題ではありません)、新しいファイルに(onCompletion後に)書き込み、その後一時ファイルを削除します。
これまでに 3 つの異なる Android バージョンとデバイスでテストしました: - Android 2.2 を搭載したタブレット - Android 4.2 を搭載した Nexus 7 - Android 4.1.2 を搭載した HTC Sensation
それは単に機能しませんでした。ビデオを含む最初の100ミリ秒を再生します。音が出て、ポップアップがポップアップし、logcat が教えてくれます:
12-19 14:33:48.074: W/MediaPlayer(5560): info/warning (3, 0)
12-19 14:33:48.074: I/MediaPlayer(5560): Info (3,0)
12-19 14:33:48.394: E/MediaPlayer(5560): error (1, -1007)
12-19 14:33:48.394: E/MediaPlayer(5560): Error (1,-1007)
しかし、これが何を意味するのか、私にはまったくわかりません。インターネット全体で検索しましたが、見つかったのは「WORLD_READABLEを使用すると問題が解決します!」というものだけでした。または「ファイル記述子を使用するだけで、パーミッションを気にしないようです!」。しかし、私にとってはうまくいきません。
助けていただければ幸いです。
編集:
FileInputStream fid = new FileInputStream((getFilesDir() + "/"
+ Downloader.FILENAME + ext)));
mVideoView.setVideoFD(fid.getFD());
これが、File Descriptor を MediaPlayer に追加する方法です。
ファイルを再エンコードした後、これが得られます。
12-19 15:06:45.664: E/MediaPlayer(7616): Unable to to create media player
12-19 15:06:45.664: W/System.err(7616): java.io.IOException: setDataSourceFD failed.: status=0x80000000
12-19 15:06:45.664: W/System.err(7616): at android.media.MediaPlayer.setDataSource(Native Method)
12-19 15:06:45.664: W/System.err(7616): at android.media.MediaPlayer.setDataSource(MediaPlayer.java:976)
これは、Webサーバーからダウンロードする方法です。
Log.i(TAG, "Opening remote connection to " + mFile.getHost()
+ mFile.getPath());
URLConnection c = mFile.openConnection();
c.connect();
final long lastModified = c.getLastModified();
final String mExtension;
if (c.getHeaderField("Content-Disposition") != null) {
final String mFilename = c
.getHeaderField("Content-Disposition").split("=")[1];
Log.i("Downloader", "Filename is " + mFilename
+ ", split length is " + mFilename.split("\\.").length);
mExtension = mFilename.split("\\.")[mFilename
.split("\\.").length - 1];
} else {
mExtension = "mp4";
}
InputStream is = c.getInputStream();
Log.i(TAG, "Creating temporary local file");
// create local temporary file
FileOutputStream fos = mContext.openFileOutput(TMP_FILENAME,
Context.MODE_WORLD_READABLE);
// start reading
byte[] buf = new byte[BUFFER_SIZE];
int bytesRead = 0;
int curRead = 0;
Log.i(TAG, "Starting download.. to " + TMP_FILENAME);
if (mDownloadChangeListener != null)
mDownloadChangeListener.onDownloadStart();
while ((curRead = is.read(buf)) > -1) {
fos.write(buf);
bytesRead += curRead;
}
Log.i(TAG, "Read " + bytesRead + " bytes in total.");
Log.i(TAG, "Download finished!");
// end of stream, tell app to rename file.
if (mDownloadChangeListener != null)
mDownloadChangeListener.onDownloadFinished(TMP_FILENAME,
mExtension);
is.close();
fos.close();
ダウンロード後、onDownloadFinished リスナーで次のコードを実行します。
FileInputStream fis = openFileInput(Downloader.TMP_FILENAME);
FileOutputStream fos = openFileOutput(Downloader.FILENAME
+ ext, Context.MODE_WORLD_WRITEABLE);
byte[] buf = new byte[2048];
while (fis.read(buf) > 0) {
fos.write(buf);
}
fis.close();
fos.close();
deleteFile(Downloader.TMP_FILENAME);
Log.d(TAG, "Replacement done!");
何がうまくいかない可能性がありますか?私の唯一の他の考えは、それが内部ストレージのせいだということです..しかし、エラーメッセージは何か他のことを言っていますか?