I have some large files (images and video) which I need to store in a content provider. The android documentation indicates...
If you are exposing byt
There is a problem writing files though. How does the content provider know then the write is complete?
When closing an OutputStream obtained via a ContentResolver.openOutputStream() I wish for the corresponding (custom) ContentProvider to perform some custom action. Presently I am using creating a FileObserver on a temporary file but it seems like the wrong way to get it done. My first thought was to subtype the ParcelFileDescriptor produced by the ContentProvider.openFile() method but that didn't seem to work for me.
That being said there were some errors which resulted in the MyParcelFileDescriptor usage.
Bottom line, is there any interaction between the file objects as seen by the ContentResolver and those of the ContentProvider?
The solution phreed gives in the bottom half of question is basically correct. I'll try to add some more details here.
When you do getContentResolver().openInputStream(...)
, content resolver will go to your content provider and call its openFile
method. This is how the openFile
looks in ContentProvider.java:
public ParcelFileDescriptor openFile(Uri uri, String mode)
throws FileNotFoundException {
throw new FileNotFoundException("No files supported by provider at "
+ uri);
}
So this explains where the "No files supported ..." error exactly comes from! You get around this by overriding openFile
method in your subclass and providing your own implementation. It's neat: you get perfect control of where your files get placed when any client does openInputStream
or openOutputStream
.
Code sample in phreed's question gives a hint how the implementation could look like. Here's my slightly modified version which also creates directories and files as needed. I'm novice at this stuff so this might not be the optimal way of doing things, but it gives an idea. For one thing, it should probably check if external storage is available.
@Override
public ParcelFileDescriptor openFile(Uri uri, String mode) throws FileNotFoundException {
File root = new File(Environment.getExternalStorageDirectory(),
"/Android/data/com.example.myapp/cache");
root.mkdirs();
File path = new File(root, uri.getEncodedPath());
// So, if the uri was content://com.example.myapp/some/data.xml,
// we'll end up accessing /Android/data/com.example.myapp/cache/some/data.xml
int imode = 0;
if (mode.contains("w")) {
imode |= ParcelFileDescriptor.MODE_WRITE_ONLY;
if (!path.exists()) {
try {
path.createNewFile();
} catch (IOException e) {
// TODO decide what to do about it, whom to notify...
e.printStackTrace();
}
}
}
if (mode.contains("r")) imode |= ParcelFileDescriptor.MODE_READ_ONLY;
if (mode.contains("+")) imode |= ParcelFileDescriptor.MODE_APPEND;
return ParcelFileDescriptor.open(path, imode);
}
Android provides the helper method openFileHelper() that makes implementing the openFile() method very easy. All you have to do, to use this method, is to provide the location of the file in a column named “_data“.
@Override
public ParcelFileDescriptor openFile(Uri uri, String mode)
throws FileNotFoundException {
if (URI_MATCHER.match(uri) != PHOTO_ID) {
throw new IllegalArgumentException
("URI invalid. Use an id-based URI only.");
}
return openFileHelper(uri, mode);
}
Click here for detail