7

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.

私は何をすべきか?

4

2 に答える 2

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());
    }
}

次に、マニフェストでコンテンツプロバイダーへの参照が必要です。私の場合は

<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 db を指す ID を格納するエンティティであり、データをフェッチしてオブジェクトに再度データを入力するクラスを使用します (_mediaData を使用)。必要に応じてコードを変更するだけです。

このようにして、アプリケーションであなたの問題を正確に解決しました

于 2012-12-18T14:36:46.210 に答える
0

このフォールバックは必要ないことを理解しました。Google Playを搭載したデバイスでは、で少なくとも2GBが使用可能であることが保証されていますEnvironment.getExternalStorageDirectory()

Galaxy Nexusでは、これは外部としてマークされた内部メモリ上のパーティションだと思います。利用できない場合は警告を表示します。

于 2012-12-15T09:11:11.123 に答える