แกลเลอรี Android บน Android 4.4 (KitKat) จะส่งคืน URI ที่แตกต่างกันสำหรับเจตนา EXACT_GET_CONTENT


214

ก่อน KitKat (หรือก่อนแกลเลอรีใหม่) Intent.ACTION_GET_CONTENTURI ที่ส่งกลับจะเป็นแบบนี้

เนื้อหา: // สื่อ / ภายนอก / images / สื่อ / 3951

การใช้ContentResolverและการสอบถามเพื่อ MediaStore.Images.Media.DATAคืนค่า URL ไฟล์

ใน KitKat อย่างไรก็ตามแกลเลอรีจะส่งคืน URI (ผ่าน "Last") ดังนี้:

เนื้อหา: //com.android.providers.media.documents/document/image: 3951

ฉันจะจัดการสิ่งนี้ได้อย่างไร


21
ฉันจะค้นหาวิธีใช้เนื้อหาที่ไม่ต้องการการเข้าถึงไฟล์โดยตรง ตัวอย่างเช่นUriควรเปิดเป็นสตรีมผ่านContentResolverได้ ฉันกังวลมานานเกี่ยวกับแอพที่คิดว่าcontent:// Uriไฟล์ที่แสดงถึงไฟล์นั้นสามารถแปลงเป็น a Fileได้เสมอ
CommonsWare

1
@ CommonsWare หากฉันต้องการบันทึกเส้นทางภาพใน sqlite db เพื่อให้สามารถเปิดได้ในภายหลังฉันควรบันทึก URI หรือเส้นทางไฟล์แบบสัมบูรณ์หรือไม่
Snake

2
@CommonsWare ฉันเห็นด้วยกับความกังวลใจของคุณ :-) อย่างไรก็ตามฉันต้องสามารถส่งชื่อไฟล์ (สำหรับรูปภาพ) ไปยังโค้ดเนมได้ ทางออกคือการคัดลอกข้อมูลที่ได้รับโดยใช้InputStreamบนContentResolverไปยังสถานที่ที่กำหนดไว้ล่วงหน้าเพื่อให้มีชื่อไฟล์ที่รู้จักกัน อย่างไรก็ตามมันฟังดูไร้ประโยชน์สำหรับฉัน ข้อเสนอแนะอื่น ๆ ?
darrenp

2
@darrenp: อืมมมม ... เขียนรหัสเนทีฟเพื่อทำงานกับInputStreamJNI มากกว่าหรือเปล่า มีตัวเลือกมากมายสำหรับคุณที่น่าเสียดาย
CommonsWare

1
มีประโยชน์ที่จะรู้ ขอบคุณสำหรับคำตอบของคุณ ฉันพบว่าตอนนี้เราส่งภาพไปยัง C ++ ในหน่วยความจำมากกว่าผ่านไฟล์ดังนั้นตอนนี้เราจึงสามารถใช้InputStreamไฟล์แทน (ซึ่งยอดเยี่ยม) เฉพาะการอ่านแท็ก EXIF เป็นเรื่องยุ่งยากเล็กน้อยและต้องใช้ห้องสมุด Drew Noakes' ขอบคุณมากสำหรับความคิดเห็นของคุณ
darrenp

คำตอบ:


108

ลองสิ่งนี้:

if (Build.VERSION.SDK_INT <19){
    Intent intent = new Intent(); 
    intent.setType("image/jpeg");
    intent.setAction(Intent.ACTION_GET_CONTENT);
    startActivityForResult(Intent.createChooser(intent, getResources().getString(R.string.select_picture)),GALLERY_INTENT_CALLED);
} else {
    Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
    intent.addCategory(Intent.CATEGORY_OPENABLE);
    intent.setType("image/jpeg");
    startActivityForResult(intent, GALLERY_KITKAT_INTENT_CALLED);
}

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (resultCode != Activity.RESULT_OK) return;
    if (null == data) return;
    Uri originalUri = null;
    if (requestCode == GALLERY_INTENT_CALLED) {
        originalUri = data.getData();
    } else if (requestCode == GALLERY_KITKAT_INTENT_CALLED) {
        originalUri = data.getData();
        final int takeFlags = data.getFlags()
                & (Intent.FLAG_GRANT_READ_URI_PERMISSION
                | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        // Check for the freshest data.
        getContentResolver().takePersistableUriPermission(originalUri, takeFlags);
    }

    loadSomeStreamAsynkTask(originalUri);

}

อาจจำเป็นต้องใช้

@SuppressLint ( "NewApi")

สำหรับ

takePersistableUriPermission


1
คุณคิดจะทำอย่างละเอียดว่ารหัส KitKat กำลังทำอะไรอยู่? สิ่งนี้ต้องการการอนุญาตใหม่หรือไม่? รหัส KitKat ก่อนทำงานให้ฉันใน KitKat ด้วย เหตุใดฉันจึงเลือกใช้รหัสอื่นสำหรับ KitKat ขอบคุณ
Michael Greifeneder

67
ดูเหมือนว่าเราไม่สามารถรับเส้นทางจาก sdks uri ใหม่ นอกจากนี้มันเป็นความอัปยศของ Google ที่ทำให้เกิดการเปลี่ยนแปลงแบบนี้โดยไม่มีเอกสารประกอบและการประกาศที่เหมาะสม
user65721

1
คุณสามารถอธิบายวิธีรับ URL ไฟล์ได้อย่างไร ฉันต้องการได้รับเส้นทางที่แท้จริงใน sdcard ตัวอย่างเช่นหากเป็นรูปภาพฉันต้องการขอรับเส้นทางนี้ /storage/sdcard0/DCIM/Camera/IMG_20131118_153817_119.jpg แทนที่จะเป็นเอกสาร Uri
Álvaro

4
จาก KitKat docs ( developer.android.com/about/versions/ ...... ) สิ่งนี้อาจไม่ใช่สิ่งที่ OP ต้องการเว้นแต่เขาจะตั้งใจใช้ / แก้ไขเอกสารที่เป็นของแอปพลิเคชันอื่น หาก OP ต้องการคัดลอกหรือทำสิ่งต่าง ๆ ในลักษณะที่สอดคล้องกับรุ่นเก่าคำตอบโดย @voytez จะเหมาะสมกว่า
โคลินเอ็ม

8
มันไม่ทำงานสำหรับฉัน ฉันได้รับข้อยกเว้นต่อไปนี้ (ในสต็อก 4.4.2): E / AndroidRuntime (29204): เกิดจาก: java.lang.SecurityException: ธงที่ร้องขอ 0x1 แต่อนุญาตเฉพาะ 0x0
รัสเซลล์สจ๊วต

177

สิ่งนี้ไม่ต้องการการอนุญาตพิเศษและทำงานกับ Storage Access Framework เช่นเดียวกับContentProviderรูปแบบที่ไม่เป็นทางการ(เส้นทางของไฟล์ใน_dataฟิลด์)

/**
 * Get a file path from a Uri. This will get the the path for Storage Access
 * Framework Documents, as well as the _data field for the MediaStore and
 * other file-based ContentProviders.
 *
 * @param context The context.
 * @param uri The Uri to query.
 * @author paulburke
 */
public static String getPath(final Context context, final Uri uri) {

    final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;

    // DocumentProvider
    if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
        // ExternalStorageProvider
        if (isExternalStorageDocument(uri)) {
            final String docId = DocumentsContract.getDocumentId(uri);
            final String[] split = docId.split(":");
            final String type = split[0];

            if ("primary".equalsIgnoreCase(type)) {
                return Environment.getExternalStorageDirectory() + "/" + split[1];
            }

            // TODO handle non-primary volumes
        }
        // DownloadsProvider
        else if (isDownloadsDocument(uri)) {

            final String id = DocumentsContract.getDocumentId(uri);
            final Uri contentUri = ContentUris.withAppendedId(
                    Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));

            return getDataColumn(context, contentUri, null, null);
        }
        // MediaProvider
        else if (isMediaDocument(uri)) {
            final String docId = DocumentsContract.getDocumentId(uri);
            final String[] split = docId.split(":");
            final String type = split[0];

            Uri contentUri = null;
            if ("image".equals(type)) {
                contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
            } else if ("video".equals(type)) {
                contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
            } else if ("audio".equals(type)) {
                contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
            }

            final String selection = "_id=?";
            final String[] selectionArgs = new String[] {
                    split[1]
            };

            return getDataColumn(context, contentUri, selection, selectionArgs);
        }
    }
    // MediaStore (and general)
    else if ("content".equalsIgnoreCase(uri.getScheme())) {

        // Return the remote address
        if (isGooglePhotosUri(uri))
            return uri.getLastPathSegment();

        return getDataColumn(context, uri, null, null);
    }
    // File
    else if ("file".equalsIgnoreCase(uri.getScheme())) {
        return uri.getPath();
    }

    return null;
}

/**
 * Get the value of the data column for this Uri. This is useful for
 * MediaStore Uris, and other file-based ContentProviders.
 *
 * @param context The context.
 * @param uri The Uri to query.
 * @param selection (Optional) Filter used in the query.
 * @param selectionArgs (Optional) Selection arguments used in the query.
 * @return The value of the _data column, which is typically a file path.
 */
public static String getDataColumn(Context context, Uri uri, String selection,
        String[] selectionArgs) {

    Cursor cursor = null;
    final String column = "_data";
    final String[] projection = {
            column
    };

    try {
        cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
                null);
        if (cursor != null && cursor.moveToFirst()) {
            final int index = cursor.getColumnIndexOrThrow(column);
            return cursor.getString(index);
        }
    } finally {
        if (cursor != null)
            cursor.close();
    }
    return null;
}


/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is ExternalStorageProvider.
 */
public static boolean isExternalStorageDocument(Uri uri) {
    return "com.android.externalstorage.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is DownloadsProvider.
 */
public static boolean isDownloadsDocument(Uri uri) {
    return "com.android.providers.downloads.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is MediaProvider.
 */
public static boolean isMediaDocument(Uri uri) {
    return "com.android.providers.media.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is Google Photos.
 */
public static boolean isGooglePhotosUri(Uri uri) {
    return "com.google.android.apps.photos.content".equals(uri.getAuthority());
}

ดูรุ่นขึ้นไปวันที่ของวิธีการนี้ที่นี่


2
สิ่งนี้ทำงานได้อย่างยอดเยี่ยมบน 4.4 Nexus 5 Documents UI และอุปกรณ์ KitKat pref อื่น ๆ ที่ใช้แอปแกลเลอรีมาตรฐานขอบคุณ Paul!
Josh

1
ขอบคุณสำหรับสิ่งนี้ฉันใช้เวลานานมากในการเข้าถึง SDK 19 !! ปัญหาของฉันคืออุปกรณ์ของฉันใช้ Google ไดรฟ์เป็นเบราว์เซอร์ไฟล์ หากไฟล์อยู่ในพา ธ อิมเมจของอุปกรณ์นั้นใช้ได้ แต่หากไฟล์อยู่ในไดรฟ์ไฟล์นั้นจะไม่เปิด บางทีฉันอาจต้องดูที่การจัดการกับรูปภาพเริ่มต้นจาก google ไดรฟ์ สิ่งที่เป็นแอพของฉันถูกเขียนขึ้นเพื่อใช้เส้นทางของไฟล์และรับภาพโดยใช้ insampling ...
RuAware

2
@RuAware เมื่อคุณเลือกไฟล์ไดรฟ์จะช่วยให้กลับมาและAuthority: com.google.android.apps.docs.storage Segments: [document, acc=1;doc=667]ฉันไม่แน่ใจ แต่คิดว่าdocค่าเป็นUriID ที่คุณสามารถค้นหาได้ คุณจะต้องมีแนวโน้มสิทธิ์ในการตั้งค่าได้ตามรายละเอียดใน "อนุญาตให้แอปของคุณบน Android" ที่นี่: developers.google.com/drive/integrate-android-ui โปรดอัปเดตที่นี่หากคุณเข้าใจ
Paul Burke

30
นี่มันช่างน่ากลัวจริงๆ! คุณไม่ควรเผยแพร่รหัสที่ "โกง" อย่างนี้ต่อไป รองรับเฉพาะแอพต้นทางที่คุณรู้จักรูปแบบและจุดรวมของรุ่นผู้ให้บริการเอกสารคือการสนับสนุนแหล่งข้อมูลโดยพลการ
j__m

2
_dataจะไม่ทำงานเมื่อ ContentProvider ไม่สนับสนุน ขอแนะนำให้ทำตามคำแนะนำ @CommonsWareและไม่ใช้เส้นทางไฟล์แบบเต็มอีกต่อไปเพราะอาจเป็นไฟล์ในระบบคลาวด์ Dropbox แทนไฟล์จริง
soshial

67

มีปัญหาเดียวกันให้ลองวิธีการแก้ปัญหาข้างต้น แต่มันก็ใช้งานได้ตามปกติด้วยเหตุผลบางอย่างที่ฉันได้รับการปฏิเสธการอนุญาตจากผู้ให้บริการเนื้อหาของ Uri สำหรับภาพบางภาพแม้ว่าฉันจะได้android.permission.MANAGE_DOCUMENTSรับการอนุญาตอย่างถูกต้องแล้วก็ตาม

ยังพบวิธีแก้ไขปัญหาอื่น ๆ ซึ่งเป็นการบังคับให้เปิดแกลเลอรี่รูปภาพแทนการดูเอกสาร KITKAT ด้วย:

// KITKAT

i = new Intent(Intent.ACTION_PICK,android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
    startActivityForResult(i, CHOOSE_IMAGE_REQUEST);

แล้วโหลดภาพ:

Uri selectedImageURI = data.getData();
input = c.getContentResolver().openInputStream(selectedImageURI);
                BitmapFactory.decodeStream(input , null, opts);

แก้ไข

ACTION_OPEN_DOCUMENT อาจต้องการให้คุณคงสถานะแฟล็กการอนุญาต ฯลฯ และโดยทั่วไปแล้วจะส่งผลให้เกิดข้อยกเว้นด้านความปลอดภัย ...

โซลูชันอื่น ๆ คือการใช้การACTION_GET_CONTENTรวมกับc.getContentResolver().openInputStream(selectedImageURI)ซึ่งจะทำงานทั้งใน pre-KK และ KK Kitkat จะใช้มุมมองเอกสารใหม่จากนั้นโซลูชันนี้จะทำงานร่วมกับแอพทั้งหมดเช่นรูปภาพ, คลังภาพ, File Explorer, Dropbox, Google Drive ฯลฯ ) แต่จำไว้ว่าเมื่อใช้โซลูชันนี้คุณจะต้องสร้างภาพในของคุณonActivityResult()และเก็บไว้ใน ตัวอย่างการ์ด SD การสร้างภาพนี้ใหม่จาก uri ที่บันทึกไว้ในการเปิดตัวแอปครั้งต่อไปจะทำให้เกิดข้อยกเว้นด้านความปลอดภัยบนตัวแก้ไขเนื้อหาแม้ว่าคุณจะเพิ่มการอนุญาตตามที่อธิบายไว้ในเอกสารของ Google API (นั่นคือสิ่งที่เกิดขึ้นเมื่อฉันทำการทดสอบ)

นอกจากนี้หลักเกณฑ์สำหรับนักพัฒนา Android ของ Android แนะนำ:

ACTION_OPEN_DOCUMENT ไม่ได้มีวัตถุประสงค์เพื่อทดแทน ACTION_GET_CONTENT สิ่งที่คุณควรใช้ขึ้นอยู่กับความต้องการของแอพของคุณ:

ใช้ ACTION_GET_CONTENT หากคุณต้องการให้แอปอ่าน / นำเข้าข้อมูล ด้วยวิธีนี้แอพจะนำเข้าสำเนาของข้อมูลเช่นไฟล์ภาพ

ใช้ ACTION_OPEN_DOCUMENT หากคุณต้องการให้แอปของคุณมีการเข้าถึงเอกสารระยะยาวและถาวรจากผู้ให้บริการเอกสาร ตัวอย่างจะเป็นแอปแก้ไขรูปภาพที่ให้ผู้ใช้แก้ไขรูปภาพที่เก็บไว้ในผู้ให้บริการเอกสาร


1
คำตอบนี้มีข้อมูลที่ถูกต้องสำหรับวัตถุประสงค์ของฉัน การใช้ ACTION_PICK และ EXTERNAL_CONTENT_URI แบบมีเงื่อนไขบน KitKat นั้นให้ความสามารถในการรับข้อมูลเมตาเกี่ยวกับรูปภาพในแกลเลอรีผ่าน ContentResolver เช่นเดียวกับรุ่นเก่าโดยใช้ ACTION_GET_CONTENT
โคลินเอ็ม

@voytez, URI นี้สามารถส่งคืนข้อความของคุณเป็นเส้นทางที่แท้จริงของภาพได้หรือไม่?
งู

ฉันเชื่อว่ามันควรจะทำงานเหมือนก่อน KitKat เพราะรหัสนี้บังคับให้เปิดแกลเลอรี่รูปภาพแทนมุมมองเอกสาร KK แต่ถ้าคุณตั้งใจจะใช้มันเพื่อสร้างภาพวิธีนี้จะดีกว่าเพราะการแปลงเป็นเส้นทางจริงเป็นขั้นตอนที่ไม่จำเป็นอีกต่อไป
voytez

4
Intent.ACTION_GET_CONTENTทำงานให้ฉันเกินไปแทน อย่างไรก็ตามฉันเก็บIntent.createChooser()เสื้อคลุมไว้ใหม่Intentเพื่อให้ผู้ใช้เลือกแอปสำหรับการเรียกดูและทำงานตามที่คาดไว้ มีใครเห็นข้อเสียสำหรับโซลูชันนี้หรือไม่
lorenzo-s

1
สำหรับทุกคนที่สงสัยว่าคำพูดมาจากdeveloper.android.com/guide/topics/providers/ …
snark

38

เช่นเดียวกับ Commonsware ที่กล่าวถึงคุณไม่ควรคิดว่าสตรีมที่คุณได้รับContentResolverจะสามารถแปลงเป็นไฟล์ได้

สิ่งที่คุณควรทำจริงๆคือเปิดInputStreamจากContentProviderนั้นสร้างบิตแมปออกมา และใช้งานได้กับรุ่น 4.4 และรุ่นก่อนหน้าเช่นกัน

    //cxt -> current context

    InputStream input;
    Bitmap bmp;
    try {
        input = cxt.getContentResolver().openInputStream(fileUri);
        bmp = BitmapFactory.decodeStream(input);
    } catch (FileNotFoundException e1) {

    }

แน่นอนถ้าคุณจัดการกับภาพขนาดใหญ่คุณควรโหลดภาพเหล่านั้นด้วยความเหมาะสมinSampleSize: http://developer.android.com/training/displaying-bitmaps/load-bitmap.html แต่นั่นเป็นหัวข้ออื่น


นี่ไม่ทำงานสำหรับฉันกับ Nexus 4 ที่กำลังใช้งาน Kitkat แต่เป็นกับ Galaxy S3 ที่ใช้ 4.1.2
Dan2552

@ Dan2552 ส่วนไหนไม่ทำงาน? คุณได้รับข้อยกเว้นหรือไม่?
Michał K

กลับกลายเป็นว่าฉันใช้การเรียกเจตนาที่ไม่ถูกต้องไปยังแกลเลอรี ฉันใช้อันที่ใช้กับเอกสารประเภทใดก็ได้ แต่มีตัวกรองนามสกุลไฟล์
Dan2552

2
ช่างเป็นคำตอบที่เรียบง่ายมากขอบคุณ! สำหรับคนอื่น ๆ ที่ทำตามคำตอบนี้ 'cxt' หมายถึงบริบทปัจจุบันและมักจะเป็น 'นี้'
thomasforth

1
นี่อาจหมายความว่าไฟล์ไม่ได้อยู่ที่นั่น URI ดูเหมือนว่าตกลง
Michał K

33

ฉันเชื่อว่าคำตอบที่โพสต์แล้วควรทำให้ผู้คนไปในทิศทางที่ถูกต้อง อย่างไรก็ตามนี่คือสิ่งที่ฉันทำที่เหมาะสมสำหรับรหัสดั้งเดิมที่ฉันอัปเดต รหัสดั้งเดิมใช้ URI จากแกลเลอรีเพื่อเปลี่ยนแล้วบันทึกภาพ

ก่อนหน้า 4.4 (และไดรฟ์ Google) URIs จะมีลักษณะเช่นนี้: content: // media / external / images / media / 41

ตามที่ระบุไว้ในคำถามพวกเขามักจะมีลักษณะเช่นนี้: content: //com.android.providers.media.documents/document/image: 3951

เนื่องจากฉันต้องการความสามารถในการบันทึกภาพและไม่รบกวนรหัสที่มีอยู่แล้วฉันเพิ่งคัดลอก URI จากแกลเลอรี่ลงในโฟลเดอร์ข้อมูลของแอป จากนั้นสร้าง URI ใหม่จากไฟล์ภาพที่บันทึกไว้ในโฟลเดอร์ข้อมูล

นี่คือแนวคิด:

Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("image/*");
startActivityForResult(intent), CHOOSE_IMAGE_REQUEST);

public void onActivityResult(int requestCode, int resultCode, Intent data) {

    super.onActivityResult(requestCode, resultCode, data);

    File tempFile = new File(this.getFilesDir().getAbsolutePath(), "temp_image");

    //Copy URI contents into temporary file.
    try {
        tempFile.createNewFile();
        copyAndClose(this.getContentResolver().openInputStream(data.getData()),new FileOutputStream(tempFile));
    }
    catch (IOException e) {
        //Log Error
    }

    //Now fetch the new URI
    Uri newUri = Uri.fromFile(tempFile);

    /* Use new URI object just like you used to */
 }

หมายเหตุ - copyAndClose () เพียงทำ I / O ไฟล์เพื่อคัดลอก InputStream ลงใน FileOutputStream รหัสไม่ได้โพสต์


ค่อนข้างฉลาด ฉันก็ต้องการไฟล์ uri จริง
GreaterKing

คุณคือฮีโร่ของฉันนี่คือสิ่งที่ฉันต้องการ! ใช้งานได้ดีสำหรับไฟล์ Google ไดรฟ์เป็นอย่างดี
Mishkin

คิดนอกกรอบใช่มั้ย : D รหัสนี้ทำงานเหมือนที่ฉันคาดหวัง
WeirdElfB0y

2
โพสต์รหัสสำหรับ copyAnd ปิดคำตอบยังไม่สมบูรณ์
Bartek Pacia

24

แค่อยากจะบอกว่าคำตอบนี้ยอดเยี่ยมและฉันใช้มันเป็นเวลานานโดยไม่มีปัญหา แต่เมื่อไม่นานมานี้ฉันได้พบกับปัญหาที่ DownloadsProvider ส่งคืน URIs ในรูปแบบcontent://com.android.providers.downloads.documents/document/raw%3A%2Fstorage%2Femulated%2F0%2FDownload%2Fdoc.pdfดังนั้นแอปจึงขัดข้องNumberFormatExceptionเนื่องจากไม่สามารถแยกเซกเมนต์ uri ได้นาน แต่raw:เซ็กเมนต์มี uri โดยตรงซึ่งสามารถใช้เพื่อเรียกคืนไฟล์อ้างอิง ดังนั้นฉันจึงแก้ไขโดยแทนที่isDownloadsDocument(uri) ifเนื้อหาด้วยสิ่งต่อไปนี้:

final String id = DocumentsContract.getDocumentId(uri);
if (!TextUtils.isEmpty(id)) {
if (id.startsWith("raw:")) {
    return id.replaceFirst("raw:", "");
}
try {
    final Uri contentUri = ContentUris.withAppendedId(
            Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
    return getDataColumn(context, contentUri, null, null);
} catch (NumberFormatException e) {
    Log.e("FileUtils", "Downloads provider returned unexpected uri " + uri.toString(), e);
    return null;
}
}

2
ทำงานได้สมบูรณ์แบบ! ขอบคุณ
mikemike396

8

ฉันได้รวมหลายคำตอบไว้ในโซลูชันเดียวที่ทำงานด้วยผลลัพธ์กับพา ธ ไฟล์

ประเภท Mime ไม่เกี่ยวข้องสำหรับวัตถุประสงค์ของตัวอย่าง

            Intent intent;
            if(Build.VERSION.SDK_INT >= 19){
                intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
                intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, false);
                intent.addFlags(Intent.FLAG_GRANT_PERSISTABLE_URI_PERMISSION);
            }else{
                intent = new Intent(Intent.ACTION_GET_CONTENT);
            }
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            intent.setType("application/octet-stream");
            if(isAdded()){
                startActivityForResult(intent, RESULT_CODE);
            }

ผลการจัดการ

    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
    if( requestCode == RESULT_CODE && resultCode == Activity.RESULT_OK) {
        Uri uri = data.getData();
        if (uri != null && !uri.toString().isEmpty()) {
            if(Build.VERSION.SDK_INT >= 19){
                final int takeFlags = data.getFlags() & Intent.FLAG_GRANT_READ_URI_PERMISSION;
                //noinspection ResourceType
                getActivity().getContentResolver()
                        .takePersistableUriPermission(uri, takeFlags);
            }
            String filePath = FilePickUtils.getSmartFilePath(getActivity(), uri);
            // do what you need with it...
        }
    }
}

FilePickUtils

import android.annotation.SuppressLint;
import android.content.ContentUris;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.os.Build;
import android.os.Environment;
import android.provider.DocumentsContract;
import android.provider.MediaStore;

public class FilePickUtils {
    private static String getPathDeprecated(Context ctx, Uri uri) {
        if( uri == null ) {
            return null;
        }
        String[] projection = { MediaStore.Images.Media.DATA };
        Cursor cursor = ctx.getContentResolver().query(uri, projection, null, null, null);
        if( cursor != null ){
            int column_index = cursor
                    .getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
            cursor.moveToFirst();
            return cursor.getString(column_index);
        }
        return uri.getPath();
    }

    public static String getSmartFilePath(Context ctx, Uri uri){

        if (Build.VERSION.SDK_INT < 19) {
            return getPathDeprecated(ctx, uri);
        }
        return  FilePickUtils.getPath(ctx, uri);
    }

    @SuppressLint("NewApi")
    public static String getPath(final Context context, final Uri uri) {
        final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
        // DocumentProvider
        if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
            // ExternalStorageProvider
            if (isExternalStorageDocument(uri)) {
                final String docId = DocumentsContract.getDocumentId(uri);
                final String[] split = docId.split(":");
                final String type = split[0];

                if ("primary".equalsIgnoreCase(type)) {
                    return Environment.getExternalStorageDirectory() + "/" + split[1];
                }

                // TODO handle non-primary volumes
            }
            // DownloadsProvider
            else if (isDownloadsDocument(uri)) {
                final String id = DocumentsContract.getDocumentId(uri);
                final Uri contentUri = ContentUris.withAppendedId(
                        Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));

                return getDataColumn(context, contentUri, null, null);
            }
            // MediaProvider
            else if (isMediaDocument(uri)) {
                final String docId = DocumentsContract.getDocumentId(uri);
                final String[] split = docId.split(":");
                final String type = split[0];

                Uri contentUri = null;
                if ("image".equals(type)) {
                    contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
                } else if ("video".equals(type)) {
                    contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
                } else if ("audio".equals(type)) {
                    contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
                }

                final String selection = "_id=?";
                final String[] selectionArgs = new String[] {
                        split[1]
                };

                return getDataColumn(context, contentUri, selection, selectionArgs);
            }
        }
        // MediaStore (and general)
        else if ("content".equalsIgnoreCase(uri.getScheme())) {
            return getDataColumn(context, uri, null, null);
        }
        // File
        else if ("file".equalsIgnoreCase(uri.getScheme())) {
            return uri.getPath();
        }

        return null;
    }

    /**
     * Get the value of the data column for this Uri. This is useful for
     * MediaStore Uris, and other file-based ContentProviders.
     *
     * @param context The context.
     * @param uri The Uri to query.
     * @param selection (Optional) Filter used in the query.
     * @param selectionArgs (Optional) Selection arguments used in the query.
     * @return The value of the _data column, which is typically a file path.
     */
    public static String getDataColumn(Context context, Uri uri, String selection,
                                       String[] selectionArgs) {
        Cursor cursor = null;
        final String column = "_data";
        final String[] projection = {
                column
        };

        try {
            cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
                    null);
            if (cursor != null && cursor.moveToFirst()) {
                final int column_index = cursor.getColumnIndexOrThrow(column);
                return cursor.getString(column_index);
            }
        } finally {
            if (cursor != null)
                cursor.close();
        }
        return null;
    }


    /**
     * @param uri The Uri to check.
     * @return Whether the Uri authority is ExternalStorageProvider.
     */
    public static boolean isExternalStorageDocument(Uri uri) {
        return "com.android.externalstorage.documents".equals(uri.getAuthority());
    }

    /**
     * @param uri The Uri to check.
     * @return Whether the Uri authority is DownloadsProvider.
     */
    public static boolean isDownloadsDocument(Uri uri) {
        return "com.android.providers.downloads.documents".equals(uri.getAuthority());
    }

    /**
     * @param uri The Uri to check.
     * @return Whether the Uri authority is MediaProvider.
     */
    public static boolean isMediaDocument(Uri uri) {
        return "com.android.providers.media.documents".equals(uri.getAuthority());
    }

}

ฉันกำลังเผชิญกับปัญหา .... uri.getPath () กำลังกลับมา uri ด้วย / ภายนอกและมันก็ไม่ได้กลับเส้นทาง ฉันเพิ่มสิ่งนี้ถ้า ("เนื้อหา" .equalsIgnoreCase (uri.getScheme ())) บล็อกและสิ่งนี้ใช้งานได้ดีในขณะนี้ .. คุณสามารถอธิบายสิ่งที่มันทำ
FaisalAhmed

filePath กำลังได้รับ null .. ใน uri: content: //com.android.externalstorage.documents/document/799B-1419%3AScurrency%2FScurrency_20181117_162826.png
Bhavesh Moradiya

7

คำถาม

วิธีรับพา ธ ไฟล์จริงจาก URI

ตอบ

สำหรับความรู้ของฉันเราไม่จำเป็นต้องได้รับพา ธ ไฟล์จาก URI เพราะส่วนใหญ่เราสามารถใช้ URI เพื่อทำงานให้เสร็จ (เช่น 1. รับบิตแมป 2. ส่งไฟล์ไปยังเซิร์ฟเวอร์ ฯลฯ .)

1. ส่งไปยังเซิร์ฟเวอร์

เราสามารถส่งไฟล์ไปยังเซิร์ฟเวอร์โดยตรงโดยใช้เพียง URI

ใช้ URI เราสามารถรับ InputStream ซึ่งเราสามารถส่งโดยตรงไปยังเซิร์ฟเวอร์โดยใช้ MultiPartEntity

ตัวอย่าง

/**
 * Used to form Multi Entity for a URI (URI pointing to some file, which we got from other application).
 *
 * @param uri     URI.
 * @param context Context.
 * @return Multi Part Entity.
 */
public MultipartEntity formMultiPartEntityForUri(final Uri uri, final Context context) {
    MultipartEntity multipartEntity = new MultipartEntity(HttpMultipartMode.BROWSER_COMPATIBLE, null, Charset.forName("UTF-8"));
    try {
        InputStream inputStream = mContext.getContentResolver().openInputStream(uri);
        if (inputStream != null) {
            ContentBody contentBody = new InputStreamBody(inputStream, getFileNameFromUri(uri, context));
            multipartEntity.addPart("[YOUR_KEY]", contentBody);
        }
    }
    catch (Exception exp) {
        Log.e("TAG", exp.getMessage());
    }
    return multipartEntity;
}

/**
 * Used to get a file name from a URI.
 *
 * @param uri     URI.
 * @param context Context.
 * @return File name from URI.
 */
public String getFileNameFromUri(final Uri uri, final Context context) {

    String fileName = null;
    if (uri != null) {
        // Get file name.
        // File Scheme.
        if (ContentResolver.SCHEME_FILE.equals(uri.getScheme())) {
            File file = new File(uri.getPath());
            fileName = file.getName();
        }
        // Content Scheme.
        else if (ContentResolver.SCHEME_CONTENT.equals(uri.getScheme())) {
            Cursor returnCursor =
                    context.getContentResolver().query(uri, null, null, null, null);
            if (returnCursor != null && returnCursor.moveToFirst()) {
                int nameIndex = returnCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME);
                fileName = returnCursor.getString(nameIndex);
                returnCursor.close();
            }
        }
    }
    return fileName;
}

2. รับ BitMap จาก URI

หาก URI ชี้ไปที่รูปภาพเราจะได้รับบิตแมปมิฉะนั้นจะเป็นโมฆะ:

/**
 * Used to create bitmap for the given URI.
 * <p>
 * 1. Convert the given URI to bitmap.
 * 2. Calculate ratio (depending on bitmap size) on how much we need to subSample the original bitmap.
 * 3. Create bitmap bitmap depending on the ration from URI.
 * 4. Reference - http://stackoverflow.com/questions/3879992/how-to-get-bitmap-from-an-uri
 *
 * @param context       Context.
 * @param uri           URI to the file.
 * @param bitmapSize Bitmap size required in PX.
 * @return Bitmap bitmap created for the given URI.
 * @throws IOException
 */
public static Bitmap createBitmapFromUri(final Context context, Uri uri, final int bitmapSize) throws IOException {

    // 1. Convert the given URI to bitmap.
    InputStream input = context.getContentResolver().openInputStream(uri);
    BitmapFactory.Options onlyBoundsOptions = new BitmapFactory.Options();
    onlyBoundsOptions.inJustDecodeBounds = true;
    onlyBoundsOptions.inDither = true;//optional
    onlyBoundsOptions.inPreferredConfig = Bitmap.Config.ARGB_8888;//optional
    BitmapFactory.decodeStream(input, null, onlyBoundsOptions);
    input.close();
    if ((onlyBoundsOptions.outWidth == -1) || (onlyBoundsOptions.outHeight == -1)) {
        return null;
    }

    // 2. Calculate ratio.
    int originalSize = (onlyBoundsOptions.outHeight > onlyBoundsOptions.outWidth) ? onlyBoundsOptions.outHeight : onlyBoundsOptions.outWidth;
    double ratio = (originalSize > bitmapSize) ? (originalSize / bitmapSize) : 1.0;

    // 3. Create bitmap.
    BitmapFactory.Options bitmapOptions = new BitmapFactory.Options();
    bitmapOptions.inSampleSize = getPowerOfTwoForSampleRatio(ratio);
    bitmapOptions.inDither = true;//optional
    bitmapOptions.inPreferredConfig = Bitmap.Config.ARGB_8888;//optional
    input = context.getContentResolver().openInputStream(uri);
    Bitmap bitmap = BitmapFactory.decodeStream(input, null, bitmapOptions);
    input.close();

    return bitmap;
}

/**
 * For Bitmap option inSampleSize - We need to give value in power of two.
 *
 * @param ratio Ratio to be rounded of to power of two.
 * @return Ratio rounded of to nearest power of two.
 */
private static int getPowerOfTwoForSampleRatio(final double ratio) {
    int k = Integer.highestOneBit((int) Math.floor(ratio));
    if (k == 0) return 1;
    else return k;
}

ความคิดเห็น

  1. Android ไม่ได้ให้วิธีการใด ๆ ในการรับเส้นทางของไฟล์จาก URI และในคำตอบส่วนใหญ่เราได้เข้ารหัสค่าคงที่บางค่าซึ่งอาจแตกในการเปิดตัวคุณลักษณะ (ขออภัยฉันอาจผิด)
  2. ก่อนที่จะไปยังโซลูชันของการรับพา ธ ไฟล์จาก URI โดยตรงให้ลองถ้าคุณสามารถแก้กรณีการใช้งานของคุณด้วยวิธีการเริ่มต้นของ URI และ Android

การอ้างอิง

  1. https://developer.android.com/guide/topics/providers/content-provider-basics.html
  2. https://developer.android.com/reference/android/content/ContentResolver.html
  3. https://hc.apache.org/httpcomponents-client-ga/httpmime/apidocs/org/apache/http/entity/mime/content/InputStreamBody.html

ขอบคุณ. การใช้ Uri และ ContentResolver วิธีนี้ทำให้แอปพลิเคชันของฉันง่ายขึ้นอย่างมากเมื่อจัดการกับไฟล์
jacksonakj

6

ไลบรารี่ Android นี้รองรับการเปลี่ยนแปลงเคสใน KitKat (รวมถึงเวอร์ชันเก่า - 2.1+):
https://github.com/iPaulPro/aFileChooser

ใช้String path = FileUtils.getPath(context, uri)เพื่อแปลง Uri ที่คืนมาเป็นสตริงพา ธ ที่สามารถใช้งานได้กับทุก OS ดูข้อมูลเพิ่มเติมได้ที่นี่: https://stackoverflow.com/a/20559175/860488


3

สำหรับผู้ที่ยังใช้รหัสของ @Paul Burke กับ Android SDK เวอร์ชัน 23 ขึ้นไปหากโครงการของคุณพบข้อผิดพลาดโดยบอกว่าคุณไม่มี EXTERNAL_PERMISSION และคุณมั่นใจว่าคุณได้เพิ่มสิทธิ์ผู้ใช้ในไฟล์ AndroidManifest.xml แล้ว นั่นเป็นเพราะคุณอาจใช้ Android API 23 ขึ้นไปและ Google จำเป็นต้องรับประกันสิทธิ์อีกครั้งในขณะที่คุณดำเนินการเข้าถึงไฟล์ในรันไทม์

ซึ่งหมายความว่า: หากเวอร์ชั่น SDK ของคุณเป็น 23 หรือสูงกว่าคุณจะถูกขอให้อ่านและเขียนอนุญาตในขณะที่คุณเลือกไฟล์รูปภาพและต้องการทราบ URI ของไฟล์นั้น

และต่อไปนี้เป็นรหัสของฉันนอกเหนือจากโซลูชันของ Paul Burke ฉันเพิ่มรหัสเหล่านี้และโครงการของฉันเริ่มทำงานได้ดี

private static final int REQUEST_EXTERNAL_STORAGE = 1;
private static final String[] PERMISSINOS_STORAGE = {
    Manifest.permission.READ_EXTERNAL_STORAGE,
    Manifest.permission.WRITE_EXTERNAL_STORAGE
};

public static void verifyStoragePermissions(Activity activity) {
    int permission = ActivityCompat.checkSelfPermission(activity, Manifest.permission.WRITE_EXTERNAL_STORAGE);

    if (permission != PackageManager.PERMISSION_GRANTED) {
        ActivityCompat.requestPermissions(
                activity,
                PERMISSINOS_STORAGE,
                REQUEST_EXTERNAL_STORAGE
        );
    }
}

และในกิจกรรมและส่วนที่คุณขอ URI:

private void pickPhotoFromGallery() {

    CompatUtils.verifyStoragePermissions(this);
    Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
    intent.setType("image/*");
    // startActivityForResult(intent, REQUEST_PHOTO_LIBRARY);
    startActivityForResult(Intent.createChooser(intent, "选择照片"),
            REQUEST_PHOTO_LIBRARY);
}

ในกรณีของฉัน CompatUtils.java เป็นที่ที่ฉันกำหนดเมธอด VerifyStoragePermissions (เป็นประเภทสแตติกเพื่อให้ฉันสามารถโทรหาได้ภายในกิจกรรมอื่น ๆ )

นอกจากนี้ควรมีเหตุผลมากกว่านี้หากคุณสร้างสถานะ if ก่อนเพื่อดูว่าเวอร์ชัน SDK ปัจจุบันสูงกว่า 23 หรือไม่ก่อนที่คุณจะเรียกใช้วิธีการตรวจสอบ VerifyStoragePermissions


2

นี่คือสิ่งที่ฉันทำ:

Uri selectedImageURI = data.getData();    imageFile = new File(getRealPathFromURI(selectedImageURI)); 

private String getRealPathFromURI(Uri contentURI) {
  Cursor cursor = getContentResolver().query(contentURI, null, null, null, null);
  if (cursor == null) { // Source is Dropbox or other similar local file path
      return contentURI.getPath();
      } else { 
      cursor.moveToFirst(); 
      int idx = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA); 
      return cursor.getString(idx); 
  }
}

หมายเหตุ: managedQuery()วิธีการเลิกใช้ดังนั้นฉันไม่ได้ใช้มัน

คำตอบนี้มาจาก m3n0R กับคำถามที่Android ได้รับเส้นทางที่แท้จริงโดย Uri.getPath ()และฉันไม่ได้รับเครดิต ฉันแค่คิดว่าคนที่ไม่ได้แก้ปัญหานี้ยังสามารถใช้สิ่งนี้ได้


2
นี่ไม่ใช่คำตอบสำหรับแอปแกลเลอรีใหม่ (แอป "ผู้ให้บริการเอกสารมีเดีย" อย่างเคร่งครัด) ใน KitKat
นาโกย่า 0

แอพที่ผู้ถามเรียก "คลังภาพ" อาจเป็นตัวเลือกไฟล์ใหม่ใน kitkat FYI: addictivetips.com/android/…
nagoya0

ฉันทำสิ่งที่คล้ายกันและรับ IndexOutOfBound บน Nexus 5X, Android 6 ในบรรทัดนี้:cursor.getString(idx);
Osify

1

โปรดพยายามหลีกเลี่ยงการใช้วิธี takePersistableUriPermission เพราะทำให้เกิดข้อยกเว้นรันไทม์สำหรับฉัน / ** * เลือกจากแกลเลอรี่ * /

public void selectFromGallery() {
    if (Build.VERSION.SDK_INT < AppConstants.KITKAT_API_VERSION) {

        Intent intent = new Intent(); 
        intent.setType("image/*");
        intent.setAction(Intent.ACTION_GET_CONTENT);
        ((Activity)mCalledContext).startActivityForResult(intent,AppConstants.GALLERY_INTENT_CALLED);

    } else {

        Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
        intent.setType("image/*");
        ((Activity)mCalledContext).startActivityForResult(intent, AppConstants.GALLERY_AFTER_KITKAT_INTENT_CALLED);
    }
}

OnActivity สำหรับผลลัพธ์ในการจัดการข้อมูลรูปภาพ:

@Override ที่ได้รับการป้องกันเป็นโมฆะ onActivityResult (int requestCode, int resultCode, ข้อมูล Intent) {

    //gallery intent result handling before kit-kat version
    if(requestCode==AppConstants.GALLERY_INTENT_CALLED 
            && resultCode == RESULT_OK) {

        Uri selectedImage = data.getData();
        String[] filePathColumn = {MediaStore.Images.Media.DATA};
        Cursor cursor = getContentResolver().query(selectedImage,filePathColumn, null, null, null);
        cursor.moveToFirst();
        int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
        String filePath = cursor.getString(columnIndex);
        cursor.close();
        photoFile = new File(filePath);
        mImgCropping.startCropImage(photoFile,AppConstants.REQUEST_IMAGE_CROP);

    }
    //gallery intent result handling after kit-kat version
    else if (requestCode == AppConstants.GALLERY_AFTER_KITKAT_INTENT_CALLED 
            && resultCode == RESULT_OK) {

        Uri selectedImage = data.getData();
        InputStream input = null;
        OutputStream output = null;

        try {
            //converting the input stream into file to crop the 
            //selected image from sd-card.
            input = getApplicationContext().getContentResolver().openInputStream(selectedImage);
            try {
                photoFile = mImgCropping.createImageFile();
            } catch (IOException e) {
                e.printStackTrace();
            }catch(Exception e) {
                e.printStackTrace();
            }
            output = new FileOutputStream(photoFile);

            int read = 0;
            byte[] bytes = new byte[1024];

            while ((read = input.read(bytes)) != -1) {
                try {
                    output.write(bytes, 0, read);
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }


    }

คุณแสดงภาพในกรณีที่สองอยู่ที่ไหน
Quantum_VC

ขออภัยฉันไม่สามารถเพิ่มบรรทัดของรหัสนี้ในอื่น ๆ ถ้า mImgCropping.startCropImage (photoFile, AppConstants.REQUEST_IMAGE_CROP); จำเป็นต้องเรียกใช้ฟังก์ชันการนึกภาพอีกครั้งตามความต้องการในโครงการของฉัน
saranya

photoFile และ mImgCropping คือไฟล์ประเภทใด
Philip BH

1

หากใครสนใจฉันสร้าง Kotlin เวอร์ชั่นสำหรับACTION_GET_CONTENT:

var path: String = uri.path // uri = any content Uri
val databaseUri: Uri
val selection: String?
val selectionArgs: Array<String>?
if (path.contains("/document/image:")) { // files selected from "Documents"
    databaseUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI
    selection = "_id=?"
    selectionArgs = arrayOf(DocumentsContract.getDocumentId(uri).split(":")[1])
} else { // files selected from all other sources, especially on Samsung devices
    databaseUri = uri
    selection = null
    selectionArgs = null
}
try {
    val projection = arrayOf(MediaStore.Images.Media.DATA,
        MediaStore.Images.Media._ID,
        MediaStore.Images.Media.ORIENTATION,
        MediaStore.Images.Media.DATE_TAKEN) // some example data you can query
    val cursor = context.contentResolver.query(databaseUri,
        projection, selection, selectionArgs, null)
    if (cursor.moveToFirst()) {
        // do whatever you like with the data
    }
    cursor.close()
} catch (e: Exception) {
    Log.e(TAG, e.message, e)
}

ฉันแค่ต้องการรหัสทำงานของ kotlin มันใช้งานได้สำหรับฉัน ขอบคุณ
Harvi Sirja

1

ฉันลองคำตอบหลายข้อที่นี่และฉันคิดว่าฉันมีทางออกที่จะทำงานทุกครั้งและจัดการสิทธิ์เช่นกัน

มันขึ้นอยู่กับการแก้ปัญหาที่ชาญฉลาดจาก LEO โพสต์นี้ควรมีรหัสทั้งหมดที่คุณต้องใช้ในการทำงานนี้และควรทำงานกับโทรศัพท์และ Android ทุกรุ่น;)

ในการที่จะมีความสามารถในการเลือกไฟล์จากการ์ด SD คุณจะต้องมีสิ่งนี้ในรายการของคุณ:

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

ค่าคงที่:

private static final int PICK_IMAGE = 456; // Whatever number you like
public static final int MY_PERMISSIONS_REQUEST_READ_EXTERNAL = 28528; // Whatever number you like
public static final String FILE_TEMP_NAME = "temp_image"; // Whatever file name you like

ตรวจสอบการอนุญาตและเรียกใช้ IMMagePick ถ้าเป็นไปได้

if (ContextCompat.checkSelfPermission(getThis(),
        Manifest.permission.READ_EXTERNAL_STORAGE)
        != PackageManager.PERMISSION_GRANTED) {

    ActivityCompat.requestPermissions(getThis(),
            new String[]{Manifest.permission.READ_EXTERNAL_STORAGE},
            MY_PERMISSIONS_REQUEST_READ_EXTERNAL);
}
else {
    launchImagePick();
}

การตอบรับอนุญาต

@Override
public void onRequestPermissionsResult(int requestCode,
                                       @NonNull
                                         String permissions[],
                                       @NonNull
                                         int[] grantResults) {

    if (manageReadExternalPermissionResponse(this, requestCode, grantResults)) {
        launchImagePick();
    }
}

จัดการการตอบกลับการอนุญาต

public static boolean manageReadExternalPermissionResponse(final Activity activity, int requestCode, int[] grantResults) {

    if (requestCode == MY_PERMISSIONS_REQUEST_READ_EXTERNAL) {

        // If request is cancelled, the result arrays are empty.

        if (grantResults.length > 0
                && grantResults[0] == PackageManager.PERMISSION_GRANTED) {

            // Permission was granted, yay! Do the
            // contacts-related task you need to do.
            return true;

        } else if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_DENIED) {

            boolean showRationale = ActivityCompat.shouldShowRequestPermissionRationale(activity,
                    Manifest.permission.READ_EXTERNAL_STORAGE);

            if (!showRationale) {
                // The user also CHECKED "never ask again".
                // You can either enable some fall back,
                // disable features of your app
                // or open another dialog explaining
                // again the permission and directing to
                // the app setting.

            } else {
                // The user did NOT check "never ask again".
                // This is a good place to explain the user
                // why you need the permission and ask if he/she wants
                // to accept it (the rationale).
            }
        } else {
            // Permission denied, boo! Disable the
            // functionality that depends on this permission.
        }
    }
    return false;
}

เรียกใช้การเลือกรูปภาพ

private void launchImagePick() {

    Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
    intent.setType("image/*");
    intent.addCategory(Intent.CATEGORY_OPENABLE);
    startActivityForResult(intent, PICK_IMAGE);

    // see onActivityResult
}

จัดการการตอบสนองการเลือกรูปภาพ

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (requestCode == PICK_IMAGE) {

        if (resultCode == Activity.RESULT_OK) {
            if (data != null && data.getData() != null) {

                try {
                     InputStream inputStream = getContentResolver().openInputStream(data.getData())
                     if (inputStream != null) {

                        // No special persmission needed to store the file like that
                        FileOutputStream fos = openFileOutput(FILE_TEMP_NAME, Context.MODE_PRIVATE);

                        final int BUFFER_SIZE = 1 << 10 << 3; // 8 KiB buffer
                        byte[] buffer = new byte[BUFFER_SIZE];
                        int bytesRead = -1;
                        while ((bytesRead = inputStream.read(buffer)) > -1) {
                            fos.write(buffer, 0, bytesRead);
                        }
                        inputStream.close();
                        fos.close();

                        File tempImageFile = new File(getFilesDir()+"/"+FILE_TEMP_NAME);

                        // Do whatever you want with the File

                        // Delete when not needed anymore
                        deleteFile(FILE_TEMP_NAME);
                    }
                }
                catch (Exception e) {
                    e.printStackTrace();
                }
            } else {
                // Error display
            }
        } else {
            // The user did not select any image
        }
    }
}

นั่นคือทั้งหมดที่คน; สิ่งนี้ใช้ได้กับโทรศัพท์ทุกรุ่นที่ฉันมี


0

นี่คือแฮ็คทั้งหมด แต่นี่คือสิ่งที่ฉันทำ ...

ดังนั้นในขณะที่เล่นกับการตั้งค่าDocumentsProviderฉันสังเกตว่าโค้ดตัวอย่าง (ในgetDocIdForFile, ประมาณเส้น 450) สร้าง id ที่ไม่ซ้ำกันสำหรับเอกสารที่เลือกตามเส้นทาง (ไม่ซ้ำกัน) ของไฟล์เทียบกับรูทที่ระบุที่คุณให้ (นั่นคือ สิ่งที่คุณตั้งไว้mBaseDirที่บรรทัด 96)

ดังนั้น URI จึงดูเหมือนว่า:

content://com.example.provider/document/root:path/to/the/file

อย่างที่เอกสารบอกว่ามันแค่รูทเดียว (ในกรณีของฉันนั่นคือEnvironment.getExternalStorageDirectory()แต่คุณอาจใช้ที่อื่น ... จากนั้นจะใช้เส้นทางของไฟล์เริ่มต้นที่รูทและทำให้เป็น ID ที่ไม่ซ้ำกันเตรียม " root:" ดังนั้นฉัน สามารถกำหนดพา ธ โดยกำจัด"/document/root:ส่วน "จาก uri.getPath () สร้างเส้นทางไฟล์จริงโดยทำดังนี้:

public void onActivityResult(int requestCode, int resultCode, Intent data) {
// check resultcodes and such, then...
uri = data.getData();
if (uri.getAuthority().equals("com.example.provider"))  {
    String path = Environment.getExternalStorageDirectory(0.toString()
                 .concat("/")
                 .concat(uri.getPath().substring("/document/root:".length())));
    doSomethingWithThePath(path); }
else {
    // another provider (maybe a cloud-based service such as GDrive)
    // created this uri.  So handle it, or don't.  You can allow specific
    // local filesystem providers, filter non-filesystem path results, etc.
}

ฉันรู้ว่า. มันน่าละอาย แต่ใช้ได้ผล อีกครั้งนี้ขึ้นอยู่กับคุณโดยใช้ของคุณเองผู้ให้บริการเอกสารในแอปของคุณเพื่อสร้าง ID เอกสาร

(นอกจากนี้ยังมีวิธีที่ดีกว่าในการสร้างเส้นทางที่ไม่คิดว่า "/" เป็นตัวคั่นเส้นทางเป็นต้น แต่คุณจะได้รับแนวคิดนี้)


ในการตอบกลับถึงตัวฉันเองด้วยความคิดที่บ้าคลั่งยิ่งขึ้น - หากแอปของคุณจัดการfile://intent จากตัวเลือกไฟล์ภายนอกแล้วคุณสามารถตรวจสอบสิทธิ์ดังในตัวอย่างด้านบนเพื่อให้แน่ใจว่ามาจากผู้ให้บริการที่กำหนดเองของคุณ นอกจากนี้ยังใช้เส้นทางเพื่อ "สร้าง" file://เจตนาใหม่โดยใช้เส้นทางที่คุณแตกแล้วStartActivity()ปล่อยให้แอปของคุณหยิบมันขึ้นมา ฉันรู้แย่มาก
fattire

0

สิ่งนี้ใช้ได้ผลดีสำหรับฉัน:

else if(requestCode == GALLERY_ACTIVITY_NEW && resultCode == Activity.RESULT_OK)
{
    Uri uri = data.getData();
    Log.i(TAG, "old uri =  " + uri);
    dumpImageMetaData(uri);

    try {
        ParcelFileDescriptor parcelFileDescriptor =
                getContentResolver().openFileDescriptor(uri, "r");
        FileDescriptor fileDescriptor = parcelFileDescriptor.getFileDescriptor();
        Log.i(TAG, "File descriptor " + fileDescriptor.toString());

        final BitmapFactory.Options options = new BitmapFactory.Options();
        options.inJustDecodeBounds = true;
        BitmapFactory.decodeFileDescriptor(fileDescriptor, null, options);

        options.inSampleSize =
           BitmapHelper.calculateInSampleSize(options,
                                              User.PICTURE_MAX_WIDTH_IN_PIXELS,
                                              User.PICTURE_MAX_HEIGHT_IN_PIXELS);
        options.inJustDecodeBounds = false;

        Bitmap bitmap = BitmapFactory.decodeFileDescriptor(fileDescriptor, null, options);
        imageViewPic.setImageBitmap(bitmap);

        ByteArrayOutputStream stream = new ByteArrayOutputStream();
        bitmap.compress(Bitmap.CompressFormat.JPEG, 100, stream);
        // get byte array here
        byte[] picData = stream.toByteArray();
        ParseFile picFile = new ParseFile(picData);
        user.setProfilePicture(picFile);
    }
    catch(FileNotFoundException exc)
    {
        Log.i(TAG, "File not found: " + exc.toString());
    }
}

ลืม dumpImageMetaData (uri); ไม่จำเป็นเลย
Rafa

0

สร้างคำตอบของ Paul Burkeฉันประสบปัญหามากมายในการแก้ไขเส้นทาง URI ของการ์ด SD ภายนอกเนื่องจากส่วนใหญ่ของ "built-in" ที่แนะนำในฟังก์ชั่นพา ธ ส่งคืนซึ่งไม่ได้รับการแก้ไขในไฟล์

อย่างไรก็ตามนี่เป็นวิธีของฉันที่// สิ่งที่ต้องทำของเขาจัดการกับปริมาณที่ไม่เกี่ยวข้อง

String resolvedPath = "";
File[] possibleExtSdComposites = context.getExternalFilesDirs(null);
for (File f : possibleExtSdComposites) {
    // Reset final path
    resolvedPath = "";

    // Construct list of folders
    ArrayList<String> extSdSplit = new ArrayList<>(Arrays.asList(f.getPath().split("/")));

    // Look for folder "<your_application_id>"
    int idx = extSdSplit.indexOf(BuildConfig.APPLICATION_ID);

    // ASSUMPTION: Expected to be found at depth 2 (in this case ExtSdCard's root is /storage/0000-0000/) - e.g. /storage/0000-0000/Android/data/<your_application_id>/files
    ArrayList<String> hierarchyList = new ArrayList<>(extSdSplit.subList(0, idx - 2));

    // Construct list containing full possible path to the file
    hierarchyList.add(tail);
    String possibleFilePath = TextUtils.join("/", hierarchyList);

    // If file is found --> success
    if (idx != -1 && new File(possibleFilePath).exists()) {
        resolvedPath = possibleFilePath;
        break;
    }
}

if (!resolvedPath.equals("")) {
    return resolvedPath;
} else {
    return null;
}

โปรดทราบว่าขึ้นอยู่กับลำดับชั้นซึ่งอาจแตกต่างกันไปตามผู้ผลิตโทรศัพท์ทุกราย - ฉันยังไม่ได้ทดสอบพวกเขาทั้งหมด (มันใช้งานได้ดีจนถึง Xperia Z3 API 23 และ Samsung Galaxy A3 API 23)

โปรดยืนยันว่ามันทำงานได้ไม่ดีที่อื่น


-1

คำตอบ @paul burke ใช้งานได้ดีสำหรับทั้งกล้องถ่ายรูปและแกลเลอรี่รูปภาพสำหรับ API ระดับ 19 ขึ้นไป แต่มันไม่ทำงานหาก SDK ขั้นต่ำของโครงการ Android ของคุณตั้งไว้ต่ำกว่า 19 และคำตอบบางอย่างที่อ้างถึงด้านบนใช้ไม่ได้กับทั้งแกลเลอรีและ กล้อง. ฉันได้แก้ไข @paul burke code ซึ่งใช้งานได้กับระดับ API ต่ำกว่า 19. ด้านล่างคือรหัส

public static String getPath(final Context context, final Uri uri) {

    final boolean isKitKat = Build.VERSION.SDK_INT >=
                             Build.VERSION_CODES.KITKAT;
    Log.i("URI",uri+"");
    String result = uri+"";

    // DocumentProvider
    // if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
    if (isKitKat && (result.contains("media.documents"))) {

        String[] ary = result.split("/");
        int length = ary.length;
        String imgary = ary[length-1];
        final String[] dat = imgary.split("%3A");

        final String docId = dat[1];
        final String type = dat[0];

        Uri contentUri = null;
        if ("image".equals(type)) {
            contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
        }
        else if ("video".equals(type)) {
        }
        else if ("audio".equals(type)) {
        }

        final String selection = "_id=?";
        final String[] selectionArgs = new String[] {
            dat[1]
        };

        return getDataColumn(context, contentUri, selection, selectionArgs);
    }
    else
    if ("content".equalsIgnoreCase(uri.getScheme())) {
        return getDataColumn(context, uri, null, null);
    }
    // File
    else if ("file".equalsIgnoreCase(uri.getScheme())) {
        return uri.getPath();
    }

    return null;
}

public static String getDataColumn(Context context, Uri uri, String selection,
                                   String[] selectionArgs) {
    Cursor cursor = null;
    final String column = "_data";
    final String[] projection = {
            column
    };

    try {
        cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
                null);
        if (cursor != null && cursor.moveToFirst()) {
            final int column_index = cursor.getColumnIndexOrThrow(column);
            return cursor.getString(column_index);
        }
    }
    finally {
        if (cursor != null)
            cursor.close();
    }
    return null;
}

ฉันได้รับ java.lang.IllegalArgumentException: ไม่มีคอลัมน์ที่ร้องขอใด ๆ เมื่อเลือกภาพ Google Doc
dirkoneill

@dirkoneill ฉันได้รับข้อยกเว้นเดียวกัน คุณพบการแก้ไขหรือไม่?
เฮนรี่

-4

คำตอบสำหรับคำถามของคุณคือคุณต้องได้รับอนุญาต พิมพ์รหัสต่อไปนี้ในไฟล์ manifest.xml ของคุณ:

<uses-sdk  android:minSdkVersion="8"   android:targetSdkVersion="18" />
<uses-permission android:name="android.permission.READ_CONTACTS" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
<uses-permission android:name="android.permission.WRITE_OWNER_DATA"></uses-permission>
<uses-permission android:name="android.permission.READ_OWNER_DATA"></uses-permission>`

มันใช้งานได้สำหรับฉัน ...

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.