我有一个应用程序,其中设置了ImageView,可以单击以在库中打开。
默认情况下,我使用以下代码从外部存储器获取文件目录以存储我的jpeg:
File picsDir = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES),"MyCameraApp");
但是!假设外部存储没有安装或根本不存在(Galaxy Nexus),这不起作用。所以我围绕它编写了一个if语句,并将内部缓存目录作为后退。
String state = Environment.getExternalStorageState()
if(Environment.MEDIA_MOUNTED.equals(state)){
File picsDir = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES),"MyCameraApp");
}else{
context.getCacheDir();
}
图像在ImageView中显示正常,但在我的意图启动时却没有显示。
Intent intent = new Intent();
intent.setAction(android.content.Intent.ACTION_VIEW);
intent.setDataAndType(Uri.fromFile(imgFile), "image/jpeg");
startActivity(intent);
图库被加载,但显示黑屏。大概是因为画廊无法访问我的应用程序的缓存目录中的文件。
作为替代方案,我尝试使用使用MediaStore.Images.Media.INTERNAL_CONTENT_URI
的媒体内容提供商,但这会在尝试投放图片时导致错误:
java.lang.UnsupportedOperationException: Writing to internal storage is not supported.
我该怎么办?
答案 0 :(得分:3)
我认为这里的问题是你正在尝试使用图库打开保存在内存私有空间中的文件(getCacheDir返回相对于应用程序的路径,只有你的应用程序可以访问该内存路径)
如果你无法保存在外部存储器中,你可以尝试保存在公共路径中(但是这样你的媒体文件可以被每个应用程序操纵,如果你卸载你的应用程序它不会清理生成的媒体你保存在那里)
如果您想使用私人内存,您可以编写ContentProvider
我编辑发布我用来完成我所说的内容提供商。 这是我的内容提供商(我刚刚发布了您需要的相关部分):
public class MediaContentProvider extends ContentProvider {
private static final String TAG = "MediaContentProvider";
// name for the provider class
public static final String AUTHORITY = "com.way.srl.HandyWay.contentproviders.media";
private MediaData _mediaData;
// UriMatcher used to match against incoming requests
private UriMatcher _uriMatcher;
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// TODO Auto-generated method stub
return 0;
}
@Override
public String getType(Uri uri) {
// TODO Auto-generated method stub
return null;
}
@Override
public Uri insert(Uri uri, ContentValues values) {
// TODO Auto-generated method stub
return null;
}
@Override
public boolean onCreate() {
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
// Add a URI to the matcher which will match against the form
// 'content://com.stephendnicholas.gmailattach.provider/*'
// and return 1 in the case that the incoming Uri matches this pattern
_uriMatcher.addURI(AUTHORITY, "*", 1);
return true;
}
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
// TODO Auto-generated method stub
return null;
}
@Override
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
// TODO Auto-generated method stub
return 0;
}
@Override
public ParcelFileDescriptor openFile(Uri uri, String mode) throws FileNotFoundException {
Log.v(TAG, "Called with uri: '" + uri + "'." + uri.getLastPathSegment());
// Check incoming Uri against the matcher
switch (_uriMatcher.match(uri)) {
// If it returns 1 - then it matches the Uri defined in onCreate
case 1:
// The desired file name is specified by the last segment of the
// path
// E.g.
// 'content://com.stephendnicholas.gmailattach.provider/Test.txt'
// Take this and build the path to the file
// String fileLocation = getContext().getCacheDir() + File.separator + uri.getLastPathSegment();
Integer mediaID = Integer.valueOf(uri.getLastPathSegment());
if (_mediaData == null) {
_mediaData = new MediaData();
}
Media m = _mediaData.get(mediaID);
// Create & return a ParcelFileDescriptor pointing to the file
// Note: I don't care what mode they ask for - they're only getting
// read only
ParcelFileDescriptor pfd = ParcelFileDescriptor.open(new File(m.filePath), ParcelFileDescriptor.MODE_READ_ONLY);
return pfd;
// Otherwise unrecognised Uri
default:
Log.v(TAG, "Unsupported uri: '" + uri + "'.");
throw new FileNotFoundException("Unsupported uri: " + uri.toString());
}
}
然后你需要在清单中提供对你的contentprovider的引用,在我的例子中它是
<provider
android:name=".contentproviders.MediaContentProvider"
android:authorities="com.way.srl.HandyWay.contentproviders.media" >
</provider>
然后像这样使用它
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.parse("content://" + MediaContentProvider.AUTHORITY + "/" + m.id), "image/jpg");
在我的情况下,m是一个实体,它存储一个指向sqlite数据库的id,我使用一个获取数据的类来再次填充对象(使用_mediaData),你只需更改代码以满足你的需求
这种方式我在我的应用程序中解决了你的问题
答案 1 :(得分:0)
我已经明白不需要这种后备。拥有Google Play的设备保证在Environment.getExternalStorageDirectory()
中至少有2 GB可用。
我想在Galaxy Nexus中,这是内部存储器上标记为外部的分区。如果它不可用,我会发出警告。