กล้อง Android: เจตนาของข้อมูลส่งคืนค่าว่าง


132

ฉันมีแอปพลิเคชั่น Android ที่มีกิจกรรมมากมาย

ในหนึ่งในนั้นฉันใช้ปุ่มที่จะเรียกกล้องของอุปกรณ์:

public void onClick(View view) {
    Intent photoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    startActivityForResult(photoIntent, IMAGE_CAPTURE);
}

ในกิจกรรมเดียวกันฉันเรียกOnActivityResultเมธอดสำหรับผลลัพธ์รูปภาพ:

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == IMAGE_CAPTURE) {
        if (resultCode == RESULT_OK) {
            Bitmap image = (Bitmap) data.getExtras().get("data");
            ImageView imageview = (ImageView) findViewById(R.id.pic);
            imageview.setImageBitmap(image);
        } else if (resultCode == RESULT_CANCELED) {
            Toast.makeText(this, "CANCELED ", Toast.LENGTH_LONG).show();
        }
    }
}

ปัญหาคือเจตนาdataเป็นโมฆะและOnActivityResultเมธอดจะเปลี่ยนไป(resultCode == RESULT_CANCELED)ที่แอปพลิเคชันโดยตรงและแอปพลิเคชันจะกลับไปที่ค่าเฉลี่ยก่อนหน้า

ฉันจะแก้ไขปัญหานี้ได้อย่างไรและหลังจากเรียกกล้องแอปพลิเคชันจะกลับสู่กิจกรรมปัจจุบันซึ่งมีสิ่งImageViewที่จะมีภาพที่ถ่าย

ขอบคุณ


ดูคำตอบนี้ได้ที่stackoverflow.com/questions/18120775/…
Praveen Sharma

คำตอบ:


218

แอปพลิเคชันกล้อง Android เริ่มต้นจะส่งคืนเจตนาที่ไม่เป็นค่าว่างเฉพาะเมื่อส่งภาพขนาดย่อใน Intent ที่ส่งคืน หากคุณส่งEXTRA_OUTPUTURI ที่จะเขียนถึงมันจะส่งคืนความnullตั้งใจและรูปภาพนั้นอยู่ใน URI ที่คุณส่งผ่าน

คุณสามารถตรวจสอบได้โดยดูที่ซอร์สโค้ดของแอปกล้องถ่ายรูปบน GitHub:

ฉันเดาว่าคุณกำลังเดินผ่านไปEXTRA_OUTPUTหรือแอพกล้องถ่ายรูปในโทรศัพท์ของคุณทำงานแตกต่างออกไป


12
ฉันจะแก้ไขฟังก์ชัน put Extra ซึ่งให้ความตั้งใจที่ไม่เป็นโมฆะในกล้องได้อย่างไร onActivityResult
Abdul Wahab

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

1
สุดยอด! ขอบคุณมากสำหรับคำอธิบาย :) +100
Muhammad Riyaz

ลิงก์ที่คุณโพสต์ไม่สามารถใช้ได้อีกต่อไป

4
เราสามารถเปลี่ยนคลาสกล้องและเปลี่ยนวิธีการเดียวเพื่ออนุญาตทั้งสองอย่างได้หรือไม่หรือยากกว่านั้น
Tanner Summers

16

ฉันพบคำตอบง่ายๆ มันได้ผล!!

private void openCameraForResult(int requestCode){
    Intent photo = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    Uri uri  = Uri.parse("file:///sdcard/photo.jpg");
    photo.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, uri);
    startActivityForResult(photo,requestCode);
}

if (requestCode == CAMERA_REQUEST_CODE) {
        if (resultCode == Activity.RESULT_OK) {
            File file = new File(Environment.getExternalStorageDirectory().getPath(), "photo.jpg");
            Uri uri = Uri.fromFile(file);
            Bitmap bitmap;
            try {
                bitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), uri);
                bitmap = cropAndScale(bitmap, 300); // if you mind scaling
                profileImageView.setImageBitmap(bitmap);
            } catch (FileNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

        }
    }

หากคุณต้องการครอบตัดและปรับขนาดภาพนี้

public static  Bitmap cropAndScale (Bitmap source, int scale){
    int factor = source.getHeight() <= source.getWidth() ? source.getHeight(): source.getWidth();
    int longer = source.getHeight() >= source.getWidth() ? source.getHeight(): source.getWidth();
    int x = source.getHeight() >= source.getWidth() ?0:(longer-factor)/2;
    int y = source.getHeight() <= source.getWidth() ?0:(longer-factor)/2;
    source = Bitmap.createBitmap(source, x, y, factor, factor);
    source = Bitmap.createScaledBitmap(source, scale, scale, false);
    return source;
}

4
ให้เป็นFileUriExposedExceptionSDK เวอร์ชันที่ทันสมัย
Souradeep Nanda

8

ฉันเคยประสบปัญหาintentนี้ไม่เป็นโมฆะ แต่intentไม่ได้รับข้อมูลที่ส่งผ่านทางนี้onActionActivit()

ป้อนคำอธิบายภาพที่นี่

นี่เป็นทางออกที่ดีกว่าโดยใช้getContentResolver () :

    private Uri imageUri;
    private ImageView myImageView;
    private Bitmap thumbnail;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

      ...
      ...    
      ...
      myImageview = (ImageView) findViewById(R.id.pic); 

      values = new ContentValues();
      values.put(MediaStore.Images.Media.TITLE, "MyPicture");
      values.put(MediaStore.Images.Media.DESCRIPTION, "Photo taken on " + System.currentTimeMillis());
      imageUri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
      Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
      intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
      startActivityForResult(intent, PICTURE_RESULT);

  }

onActivityResult()ได้รับบิตแมปที่จัดเก็บโดยgetContentResolver () :

 @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);

        if (requestCode == REQUEST_CODE_TAKE_PHOTO && resultCode == RESULT_OK) {

            Bitmap bitmap;
            try {
                bitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), imageUri);
                myImageView.setImageBitmap(bitmap);
            } catch (FileNotFoundException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        }
    }

introducir la descripción de la imagen aquí introducir la descripción de la imagen aquí

ตรวจสอบตัวอย่างของฉันใน github:

https://github.com/Jorgesys/TakePicture


6

แอพกล้องที่ใช้งานง่ายหลีกเลี่ยงปัญหาเจตนาว่าง

- รหัสที่เปลี่ยนแปลงทั้งหมดรวมอยู่ในการตอบกลับนี้ ใกล้กับการสอนของ Android

ฉันใช้เวลามากมายกับปัญหานี้ดังนั้นฉันจึงตัดสินใจสร้างบัญชีและแบ่งปันผลลัพธ์ของฉันกับคุณ

บทช่วยสอน Android อย่างเป็นทางการ"การถ่ายภาพง่ายๆ"กลับกลายเป็นว่าไม่ถือสิ่งที่สัญญาไว้ รหัสที่ให้มาใช้ไม่ได้บนอุปกรณ์ของฉัน: Samsung Galaxy S4 Mini GT-I9195 ที่ใช้ Android เวอร์ชัน 4.4.2 / KitKat / API ระดับ 19

ฉันพบว่าปัญหาหลักคือบรรทัดต่อไปนี้ในวิธีการที่เรียกใช้เมื่อถ่ายภาพ ( dispatchTakePictureIntentในบทช่วยสอน):

takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);

ส่งผลให้เจตนาถูกจับในภายหลัง onActivityResultเป็นโมฆะ

เพื่อแก้ปัญหานี้ฉันดึงแรงบันดาลใจจากการตอบกลับก่อนหน้านี้ที่นี่และโพสต์ที่เป็นประโยชน์บางส่วนใน github (ส่วนใหญ่เป็นเรื่องนี้โดย deepwinter - ต้องขอบคุณเขามากคุณอาจต้องการตรวจสอบคำตอบของเขาในโพสต์ที่เกี่ยวข้องอย่างใกล้ชิดเช่นกัน)

ตามคำแนะนำที่น่าพอใจเหล่านี้ฉันเลือกกลยุทธ์ในการลบputExtraบรรทัดที่กล่าวถึงและทำสิ่งที่สอดคล้องกันในการดึงภาพที่ถ่ายจากกล้องกลับมาภายในเมธอด onActivityResult () แทน บรรทัดของโค้ดที่ชี้ขาดเพื่อเรียกคืนบิตแมปที่เกี่ยวข้องกับรูปภาพคือ:

        Uri uri = intent.getData();
        Bitmap bitmap = null;
        try {
            bitmap = MediaStore.Images.Media.getBitmap(this.getContentResolver(), uri);
        } catch (IOException e) {
            e.printStackTrace();
        }

ฉันสร้างแอพที่เป็นแบบอย่างที่มีความสามารถในการถ่ายภาพบันทึกลงในการ์ด SD และแสดงได้ ฉันคิดว่าสิ่งนี้อาจเป็นประโยชน์สำหรับผู้ที่อยู่ในสถานการณ์เดียวกับฉันเมื่อฉันสะดุดกับปัญหานี้เนื่องจากคำแนะนำความช่วยเหลือในปัจจุบันส่วนใหญ่อ้างถึงโพสต์ github ที่ค่อนข้างกว้างขวางซึ่งทำสิ่งที่เป็นปัญหา แต่ก็ไม่ง่ายเกินไปที่จะดูแลสำหรับมือใหม่เช่น ผม. เกี่ยวกับระบบไฟล์ที่ Android Studio สร้างขึ้นตามค่าเริ่มต้นเมื่อสร้างโครงการใหม่ฉันต้องเปลี่ยนไฟล์สามไฟล์ตามวัตถุประสงค์ของฉัน:

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.example.android.simpleworkingcameraapp.MainActivity">

<Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:onClick="takePicAndDisplayIt"
    android:text="Take a pic and display it." />

<ImageView
    android:id="@+id/image1"
    android:layout_width="match_parent"
    android:layout_height="200dp" />

</LinearLayout>

MainActivity.java:

package com.example.android.simpleworkingcameraapp;

import android.content.Intent;
import android.graphics.Bitmap;
import android.media.Image;
import android.net.Uri;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import android.widget.Toast;

import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class MainActivity extends AppCompatActivity {

private ImageView image;
static final int REQUEST_TAKE_PHOTO = 1;
String mCurrentPhotoPath;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    image = (ImageView) findViewById(R.id.image1);
}

// copied from the android development pages; just added a Toast to show the storage location
private File createImageFile() throws IOException {
    // Create an image file name
    String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmm").format(new Date());
    String imageFileName = "JPEG_" + timeStamp + "_";
    File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES);
    File image = File.createTempFile(
            imageFileName,  /* prefix */
            ".jpg",         /* suffix */
            storageDir      /* directory */
    );

    // Save a file: path for use with ACTION_VIEW intents
    mCurrentPhotoPath = image.getAbsolutePath();
    Toast.makeText(this, mCurrentPhotoPath, Toast.LENGTH_LONG).show();
    return image;
}

public void takePicAndDisplayIt(View view) {
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    if (intent.resolveActivity(getPackageManager()) != null) {
        File file = null;
        try {
            file = createImageFile();
        } catch (IOException ex) {
            // Error occurred while creating the File
        }

        startActivityForResult(intent, REQUEST_TAKE_PHOTO);
    }
}

@Override
protected void onActivityResult(int requestCode, int resultcode, Intent intent) {
    if (requestCode == REQUEST_TAKE_PHOTO && resultcode == RESULT_OK) {
        Uri uri = intent.getData();
        Bitmap bitmap = null;
        try {
            bitmap = MediaStore.Images.Media.getBitmap(this.getContentResolver(), uri);
        } catch (IOException e) {
            e.printStackTrace();
        }
        image.setImageBitmap(bitmap);
    }
}
}

AndroidManifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.android.simpleworkingcameraapp">


<!--only added paragraph-->
<uses-feature
    android:name="android.hardware.camera"
    android:required="true" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />  <!-- only crucial line to add; for me it still worked without the other lines in this paragraph -->
<uses-permission android:name="android.permission.CAMERA" />


<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:roundIcon="@mipmap/ic_launcher_round"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <activity android:name=".MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
</application>

</manifest>

โปรดทราบว่าวิธีแก้ปัญหาที่ฉันพบสำหรับปัญหายังนำไปสู่ความเรียบง่ายของไฟล์รายการ android: การเปลี่ยนแปลงที่แนะนำโดยบทช่วยสอน Android ในแง่ของการเพิ่มผู้ให้บริการนั้นไม่จำเป็นอีกต่อไปเนื่องจากฉันไม่ได้ใช้ประโยชน์ใด ๆ ในรหัสจาวาของฉัน ดังนั้นจึงต้องเพิ่มบรรทัดมาตรฐานเพียงไม่กี่บรรทัด - ส่วนใหญ่เกี่ยวกับสิทธิ์ - ต้องถูกเพิ่มลงในไฟล์รายการ

นอกจากนี้ยังอาจจะมีคุณค่าที่จะชี้ให้เห็นว่า AutoImport Android สตูดิโออาจจะไม่สามารถจัดการและjava.text.SimpleDateFormat java.util.Dateฉันต้องนำเข้าทั้งสองรายการด้วยตนเอง


โซลูชันนี้ต้องได้รับอนุญาตในการจัดเก็บซึ่งไม่ใช่แนวทางที่ดีนัก
shanraisshan

4
คำถามระบุอย่างชัดเจนว่าพารามิเตอร์ Intent ชื่อ "data" ที่ส่งไปยัง onActivityResult () เป็นโมฆะในขณะที่โซลูชันของคุณจะดำเนินต่อไปและพยายามใช้สิ่งนั้น
Maks

2

อาจเป็นเพราะคุณมีอะไรแบบนี้?

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);                        
Uri fileUri =  CommonUtilities.getTBCameraOutputMediaFileUri();                  
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, fileUri);                        
startActivityForResult(takePictureIntent, 2);

อย่างไรก็ตามคุณต้องไม่ใส่เอาต์พุตพิเศษลงใน Intent เนื่องจากข้อมูลจะเข้าไปใน URI แทนที่จะเป็นตัวแปรข้อมูล ด้วยเหตุนี้คุณต้องเอาสองเส้นตรงกลางออกถึงจะได้

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(takePictureIntent, 2);

นั่นคือสิ่งที่ทำให้เกิดปัญหาสำหรับฉันหวังว่าจะช่วยได้


12
คำตอบนี้ทำให้เข้าใจผิด คุณจะได้ภาพขนาดย่อแทนภาพถ่ายขนาดเต็มโดยไม่ต้องใส่ MediaStore.EXTRA_OUTPUT ซึ่งไฮไลต์อยู่ในเอกสารและในคำตอบอื่น ๆ
user2417480

2
คำตอบนี้ทำให้เข้าใจผิดอย่างแน่นอน
IndexOutOfDevelopersException

นี่คือคำตอบที่ทำให้เข้าใจผิด
Abdulla Thanseeh

0

รหัสต่อไปนี้ใช้ได้กับฉัน:

Intent cameraIntent = new Intent(android.provider.MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(cameraIntent, 2);

และนี่คือผลลัพธ์:

protected void onActivityResult(int requestCode, int resultCode, Intent imageReturnedIntent)
{ 
    super.onActivityResult(requestCode, resultCode, imageReturnedIntent);

    if(resultCode == RESULT_OK)
    {
        Uri selectedImage = imageReturnedIntent.getData();
        ImageView photo = (ImageView) findViewById(R.id.add_contact_label_photo);
        Bitmap mBitmap = null;
        try
        {
            mBitmap = Media.getBitmap(this.getContentResolver(), selectedImage);
        }
        catch (IOException e)
        {
            e.printStackTrace();
        }
    }
}

ฉันทดสอบแล้ว แต่มันเป็นปัญหาเดียวกัน :(
soft_developer

1
ฉันได้รับimageReturnedIntent.getData()ค่าเป็นโมฆะ ฉันกำลังสร้างเจตนาเช่นเดียวกับคุณ ไม่ใส่พารามิเตอร์พิเศษใด ๆ
Jaydip Kalkani


0

รหัส Kotlin ที่เหมาะกับฉัน:

 private fun takePhotoFromCamera() {
          val intent = Intent(android.provider.MediaStore.ACTION_IMAGE_CAPTURE)
        startActivityForResult(intent, PERMISSIONS_REQUEST_TAKE_PICTURE_CAMERA)
      }

และรับผลลัพธ์:

 override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    super.onActivityResult(requestCode, resultCode, data)
 if (requestCode == PERMISSIONS_REQUEST_TAKE_PICTURE_CAMERA) {
         if (resultCode == Activity.RESULT_OK) {
           val photo: Bitmap? =  MediaStore.Images.Media.getBitmap(this.contentResolver, Uri.parse( data!!.dataString)   )
            // Do something here : set image to an ImageView or save it ..   
              imgV_pic.imageBitmap = photo 
        } else if (resultCode == Activity.RESULT_CANCELED) {
            Log.i(TAG, "Camera  , RESULT_CANCELED ")
        }

    }

}

และอย่าลืมแจ้งรหัสคำขอ:

companion object {
 const val PERMISSIONS_REQUEST_TAKE_PICTURE_CAMERA = 300
  }

0

หลังจากพยายามและศึกษามามากฉันก็สามารถคิดออก ขั้นแรกข้อมูลตัวแปรจาก Intent จะเป็นโมฆะเสมอดังนั้นการตรวจสอบ!nullจะทำให้แอปของคุณหยุดทำงานนานจนคุณส่ง URI เพื่อ startActivityForResult ทำตามตัวอย่างด้านล่าง ฉันจะใช้ Kotlin

  1. เปิดกล้องเจตนา

    fun addBathroomPhoto(){
    addbathroomphoto.setOnClickListener{
    
        request_capture_image=2
    
        var takePictureIntent:Intent?
        takePictureIntent =Intent(MediaStore.ACTION_IMAGE_CAPTURE)
        if(takePictureIntent.resolveActivity(activity?.getPackageManager()) != null){
    
            val photoFile: File? = try {
                createImageFile()
            } catch (ex: IOException) {
                // Error occurred while creating the File
    
                null
            }
    
            if (photoFile != null) {
                val photoURI: Uri = FileProvider.getUriForFile(
                    activity!!,
                    "ogavenue.ng.hotelroomkeeping.fileprovider",photoFile)
                takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT,
                    photoURI);
                startActivityForResult(takePictureIntent,
                    request_capture_image);
            }
    
    
        }
    
    }

    } `

  2. สร้าง createImageFile () แต่คุณต้องทำให้ตัวแปร imageFilePath เป็น global ตัวอย่างวิธีการสร้างอยู่ในเอกสารอย่างเป็นทางการของ Android และค่อนข้างตรงไปตรงมา

  3. รับความตั้งใจ

     override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    
    if (requestCode == 1 && resultCode == RESULT_OK) {
        add_room_photo_txt.text=""
        var myBitmap=BitmapFactory.decodeFile(imageFilePath)
        addroomphoto.setImageBitmap(myBitmap)
        var file=File(imageFilePath)
        var fis=FileInputStream(file)
        var bm = BitmapFactory.decodeStream(fis);
        roomphoto=getBytesFromBitmap(bm) }}
  4. เมธอด getBytesFromBitmap

      fun getBytesFromBitmap(bitmap:Bitmap):ByteArray{
    
      var stream=ByteArrayOutputStream()
      bitmap.compress(Bitmap.CompressFormat.JPEG, 100, stream);
      return stream.toByteArray();
      }

ฉันหวังว่านี่จะช่วยได้.


0

เมื่อเราจะจับภาพจาก Camera ใน android แล้ว Uri หรือdata.getdata()มา null เรามีสองวิธีในการแก้ไขปัญหานี้

  1. เราสามารถรับเส้นทาง Uri ได้จากภาพบิตแมป
  2. เราสามารถรับเส้นทาง Uri จากเคอร์เซอร์

ฉันจะใช้วิธีการทั้งหมดที่นี่โปรดดูและอ่านสิ่งเหล่านี้อย่างรอบคอบ: -

ก่อนอื่นฉันจะบอกวิธีรับ Uri จาก Bitmap Image: รหัสที่สมบูรณ์คือ:

ก่อนอื่นเราจะจับภาพผ่าน Intent ซึ่งจะเหมือนกันสำหรับทั้งสองวิธีดังนั้นรหัสนี้ฉันจะเขียนครั้งเดียวที่นี่เท่านั้น:

// Capture Image
captureImg.setOnClickListener(new View.OnClickListener(){
    @Override public void onClick(View view){
        Intent intent=new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        if(intent.resolveActivity(getPackageManager())!=null){
            startActivityForResult(intent,reqcode);
        }

    }
});  

ตอนนี้เราจะใช้ OnActivityResult :-( ซึ่งจะเหมือนกันสำหรับทั้ง 2 วิธีข้างต้น): -

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
    super.onActivityResult(requestCode, resultCode, data);

    if (requestCode == reqcode && resultCode == RESULT_OK)
    {

        Bitmap photo = (Bitmap) data.getExtras().get("data");
        ImageView.setImageBitmap(photo);

        // CALL THIS METHOD TO GET THE URI FROM THE BITMAP
        Uri tempUri = getImageUri(getApplicationContext(), photo);

        //Show Uri path based on Image
        Toast.makeText(LiveImage.this, "Here " + tempUri, Toast.LENGTH_LONG).show();

        //Show Uri path based on Cursor Content Resolver
        Toast.makeText(this, "Real path for URI : " + getRealPathFromURI(tempUri), Toast.LENGTH_SHORT).show();
    }
    else
    {
        Toast.makeText(this, "Failed To Capture Image", Toast.LENGTH_SHORT).show();
    }
}     

ตอนนี้เราจะสร้างวิธีการทั้งหมดข้างต้นเพื่อสร้าง Uri จากเมธอด Image และ Cursor ผ่านคลาส:

ตอนนี้เส้นทาง URI จากภาพบิตแมป

private Uri getImageUri(Context applicationContext, Bitmap photo)
{
    ByteArrayOutputStream bytes = new ByteArrayOutputStream();
    photo.compress(Bitmap.CompressFormat.JPEG, 100, bytes);
    String path = MediaStore.Images.Media.insertImage(LiveImage.this.getContentResolver(), photo, "Title", null);
    return Uri.parse(path);
}

\ Uri จากเส้นทางจริงของภาพที่บันทึกไว้

public String getRealPathFromURI(Uri uri)
{
    Cursor cursor = getContentResolver().query(uri, null, null, null, null);
    cursor.moveToFirst();
    int idx = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA);
    return cursor.getString(idx);
}

-1

เมื่อเราจับภาพจากกล้องใน Android แล้วUriหรือdata.getdata()กลายเป็นโมฆะ เรามีสองวิธีในการแก้ไขปัญหานี้

  1. ดึงเส้นทาง Uri จากภาพบิตแมป
  2. ดึงเส้นทาง Uri จากเคอร์เซอร์

นี่คือวิธีการดึงข้อมูล Uri จากภาพบิตแมป ขั้นแรกจับภาพผ่าน Intent ซึ่งจะเหมือนกันสำหรับทั้งสองวิธี:

// Capture Image
captureImg.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        if (intent.resolveActivity(getPackageManager()) != null) {
            startActivityForResult(intent, reqcode);
        }
    }
});

ตอนนี้ใช้OnActivityResultซึ่งจะเหมือนกันสำหรับทั้งสองวิธี:

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);

    if(requestCode==reqcode && resultCode==RESULT_OK)
    {
        Bitmap photo = (Bitmap) data.getExtras().get("data");
        ImageView.setImageBitmap(photo);

        // CALL THIS METHOD TO GET THE URI FROM THE BITMAP
        Uri tempUri = getImageUri(getApplicationContext(), photo);

        // Show Uri path based on Image
        Toast.makeText(LiveImage.this,"Here "+ tempUri, Toast.LENGTH_LONG).show();

        // Show Uri path based on Cursor Content Resolver
        Toast.makeText(this, "Real path for URI : "+getRealPathFromURI(tempUri), Toast.LENGTH_SHORT).show();
    }
    else
    {
        Toast.makeText(this, "Failed To Capture Image", Toast.LENGTH_SHORT).show();
    }
}

ตอนนี้สร้างวิธีการทั้งหมดข้างต้นเพื่อสร้าง Uri จากเมธอด Image และ Cursor:

เส้นทาง Uri จากภาพบิตแมป:

private Uri getImageUri(Context applicationContext, Bitmap photo) {
    ByteArrayOutputStream bytes = new ByteArrayOutputStream();
    photo.compress(Bitmap.CompressFormat.JPEG, 100, bytes);
    String path = MediaStore.Images.Media.insertImage(LiveImage.this.getContentResolver(), photo, "Title", null);
    return Uri.parse(path);
}

Uri จากเส้นทางจริงของภาพที่บันทึก:

public String getRealPathFromURI(Uri uri) {
    Cursor cursor = getContentResolver().query(uri, null, null, null, null);
    cursor.moveToFirst();
    int idx = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA);
    return cursor.getString(idx);
}

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