Skip to content

Commit 86c3dbd

Browse files
committed
Fixed to work on KitKat, by copying videos over to app's private data directory. Note there are slight security concerns when doing this, but it is the only way.
1 parent fd10762 commit 86c3dbd

File tree

2 files changed

+143
-4
lines changed

2 files changed

+143
-4
lines changed

README.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,10 @@ This plugin enables the use of normal HTML5 <video> tags for local video - other
77
Android Webview (which Cordova is based upon) limits access to local files (such as videos) and prohibits reading them, both via relative files and `file:///` URIs. Cordova Html5Video Plugin solves this by giving each of you video a `android.resource://` path and updating your `<video>` tags accordingly.
88

99
### Limitations ###
10-
For Android only. Tested on Android API 15-18 (testing for API 19 is undergoing).
10+
For Android only. Tested on Android API 15-19.
11+
12+
For API >= 19, a workaround is employed to copy the video files over to your application's data directory, as world-readable.
13+
*WARNING*: this is potentially insecure - other apps will be able to read your videos! However it is the only way to get around Chrome's strict limitations on content:// URLs. See Issue #20 for details.
1114

1215
### Install ###
1316

src/android/Html5Video.java

+139-3
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,30 @@
11
package org.apache.cordova.plugin;
22

3-
import java.util.Timer;
3+
import java.io.BufferedInputStream;
4+
import java.io.BufferedOutputStream;
5+
import java.io.File;
6+
import java.io.IOException;
47

58
import org.apache.cordova.*;
69
import org.json.JSONArray;
710
import org.json.JSONException;
811
import org.json.JSONObject;
912

13+
import android.app.Activity;
14+
import android.content.Context;
15+
import android.content.res.AssetFileDescriptor;
16+
import android.content.res.Resources.NotFoundException;
17+
import android.os.Build;
18+
import android.util.Log;
19+
1020
public class Html5Video extends CordovaPlugin {
21+
private static final String TAG = "Html5VideoCordovaPlugin";
22+
1123
public final String ACTION_INITIALIZE = "initialize";
1224
public final String ACTION_PLAY = "play";
1325

26+
FileDataStore dataStore = new FileDataStore();
27+
1428
@Override
1529
public boolean execute(String action, JSONArray args, final CallbackContext callbackContext) throws JSONException {
1630

@@ -25,8 +39,29 @@ public boolean execute(String action, JSONArray args, final CallbackContext call
2539
if (tagNames != null) {
2640
for (int i = 0; i < tagNames.length(); i++) {
2741
String[] video = videos.getString(tagNames.getString(i)).split("\\.");
28-
int videoId = this.cordova.getActivity().getResources().getIdentifier(video[0], "raw", packageName);
29-
convertedVideos.put(tagNames.getString(i), "android.resource://" + packageName + "/" + videoId);
42+
String newUrl = null;
43+
44+
// Kitkat and above don't allow loading android.resource:// URLs,
45+
// (see https://code.google.com/p/android/issues/detail?id=63033 ),
46+
// so we do a little trick.
47+
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
48+
// copy vids to global-accessible directory, then replace with that URL.
49+
// idea from this thread: https://github.com/jaeger25/Html5Video/issues/23#issuecomment-51174558
50+
String newFilePath = dataStore.getFilePath(video[0]);
51+
52+
if (newFilePath != null) {
53+
newUrl = "file://" + newFilePath;
54+
Log.d(TAG, "Using KitKat, can't load local videos. Loading from: " + newUrl);
55+
}
56+
}
57+
58+
59+
if (newUrl == null) { // do the old way if the KitKat way failed (or if we're not on KitKat)
60+
int videoId = this.cordova.getActivity().getResources().getIdentifier(video[0], "raw", packageName);
61+
newUrl = "android.resource://" + packageName + "/" + videoId;
62+
}
63+
64+
convertedVideos.put(tagNames.getString(i), newUrl);
3065

3166
LOG.d("Html5Video", "Id: " + tagNames.getString(i) + " , src: " + convertedVideos.getString(tagNames.getString(i)));
3267
}
@@ -55,4 +90,105 @@ public void run() {
5590
public Object onMessage(String id, Object data) {
5691
return super.onMessage(id, data);
5792
}
93+
94+
95+
private class FileDataStore {
96+
97+
/***
98+
* Gets the file:// URL of a video file in the res/raw directory, copying it over if necessary.
99+
*
100+
* Note: the file will be world-readable (not-recommended by Google), so be careful: other apps will be able to read your videos!
101+
*
102+
* @param rawFileName Name of the file in the raw directory.
103+
* @return The absolute path of the file, or null if copy failed.
104+
*/
105+
public String getFilePath(String rawFileName) {
106+
String fileName = "html5video_" + rawFileName; // path separators not allowed
107+
File f = cordova.getActivity().getFileStreamPath(fileName);
108+
109+
// transfer raw resrouce file into app data directory so it can be read globally.
110+
if (!f.exists()) {
111+
112+
if (!copyResourceFileToDataDir(rawFileName, fileName)) {
113+
return null;
114+
}
115+
}
116+
117+
return f.getAbsolutePath();
118+
}
119+
120+
/***
121+
* Copies a raw file from resources directory to the app's private data directory.
122+
* @param rawFileName
123+
* @param newFileName
124+
* @return True on success, false otherwise
125+
*/
126+
private boolean copyResourceFileToDataDir(String rawFileName, String newFileName) {
127+
String packageName = cordova.getActivity().getPackageName();
128+
129+
AssetFileDescriptor fd = null;
130+
BufferedInputStream in = null;
131+
BufferedOutputStream out = null;
132+
try {
133+
Activity activity = cordova.getActivity();
134+
int id = activity.getResources().getIdentifier(rawFileName, "raw", packageName);
135+
fd = activity.getResources().openRawResourceFd(id);
136+
if (fd == null) {
137+
throw new NotFoundException("Raw file not found: " + rawFileName);
138+
}
139+
140+
in = new BufferedInputStream(fd.createInputStream()); // this is auto-close!
141+
// well we need world readable because that's the only way chrome can read it!
142+
// don't worry about their scare-tactic "oooh security holes" warnings: who cares if other apps can read your videos?
143+
out = new BufferedOutputStream(activity.openFileOutput(newFileName, Context.MODE_WORLD_READABLE));
144+
145+
// copy the file over. it's 2014, we still have to do this by hand?
146+
byte buffer[] = new byte[8192];
147+
while(in.available() > 0) {
148+
int bytesRead = in.read(buffer);
149+
150+
if (bytesRead > 0) {
151+
out.write(buffer, 0, bytesRead);
152+
}
153+
}
154+
155+
in.close();
156+
out.close();
157+
158+
return true;
159+
} catch (NotFoundException e) {
160+
Log.e(TAG, "Failed to copy video file to data dir.", e);
161+
} catch (IOException e) {
162+
Log.e(TAG, "Failed to copy video file to data dir.", e);
163+
}
164+
finally {
165+
166+
try {
167+
if (fd != null) {
168+
fd.close();
169+
}
170+
} catch (IOException e) {
171+
Log.e(TAG, "Failed to fail to copy video file to data dir.", e);
172+
}
173+
174+
try {
175+
if (in != null) {
176+
in.close();
177+
}
178+
} catch (IOException e) {
179+
Log.e(TAG, "Failed to fail to copy video file to data dir.", e);
180+
}
181+
182+
try {
183+
if (out != null) {
184+
out.close();
185+
}
186+
} catch (IOException e) {
187+
Log.e(TAG, "Failed to fail to copy video file to data dir.", e);
188+
}
189+
}
190+
191+
return false;
192+
}
193+
}
58194
}

0 commit comments

Comments
 (0)