web-dev-qa-db-ja.com

Android Camera:データインテントはnullを返します

複数のアクティビティを含むAndroidアプリケーションがあります。

それらの1つでは、デバイスのカメラを呼び出すボタンを使用しています:

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がnullであり、OnActivityResultメソッドが直接(resultCode == RESULT_CANCELED)に変わり、アプリケーションが以前のアクティビティに戻ることです。

この問題を修正するにはどうすればよいですか?カメラを呼び出した後、アプリケーションは、撮影した写真を含むImageViewを含む現在のアクティビティに戻りますか?

ありがとう

127
soft_developer

デフォルトのAndroidカメラアプリケーションは、返されたIntentでサムネイルを返すときにのみ、null以外のインテントを返します。 EXTRA_OUTPUTに書き込み先のURIを渡すと、nullインテントが返され、画像は渡したURIに含まれます。

これを確認するには、GitHubでカメラアプリのソースコードを確認します。

何らかの形でEXTRA_OUTPUTを渡すか、携帯電話のカメラアプリの動作が異なると思います。

207
Brian Slesinsky

簡単な答えが見つかりました。できます!!

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;
}
15
izakos

ヌルインテントの問題を回避するシンプルな作業カメラアプリ

-この応答に含まれるすべての変更されたコード。 Androidチュートリアルに近い

私はこの問題に多くの時間を費やしてきたので、アカウントを作成し、結果をあなたと共有することにしました。

公式のAndroidチュートリアル "Taking Photos Simply" は、約束した内容を完全に守っていないことが判明しました。そこで提供されたコードは私のデバイスでは動作しませんでした:Androidバージョン4.4.2/KitKat/API Level 19を実行しているSamsung Galaxy S4 Mini GT-I9195.

主な問題は、写真をキャプチャするときに呼び出されるメソッドの次の行であることがわかりました(チュートリアルのdispatchTakePictureIntent):

takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);

その結果、onActivityResultによってキャッチされたインテントがヌルになりました。

この問題を解決するために、ここでの以前の返信とgithubのいくつかの有用な投稿から多くのインスピレーションを引き出しました(主に これはdeepwinterによる -彼に感謝します。関連 post も同様)。

これらの快適なアドバイスに従って、前述の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がデフォルトで作成するため、目的に合わせて3つのファイルを変更する必要がありました。

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チュートリアルで提案された変更は、 Javaコードでは何も使用していません。そのため、マニフェストファイルに追加する必要があるのは、ほとんどのアクセス許可に関する標準行のみです。

さらに、Android Studioの自動インポートではJava.text.SimpleDateFormatおよびJava.util.Dateを処理できない可能性があることを指摘しておくと役立つ場合があります。両方を手動でインポートする必要がありました。

4
H.G.M.

おそらくこのようなものがあったからでしょうか?

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

ただし、データはデータ変数ではなくURIに送られるため、余分な出力をインテントに入れないでください。そのため、真ん中の2行を取る必要があります。

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

それが私にとって問題の原因であり、助けてくれたことを願っています。

2

この問題が発生しました。intentはnullではありませんが、このintentを介して送信された情報はonActionActivit()で受信されません

enter image description here

これは 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

1
Jorgesys

私のために働く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
Hamed Jaliliani

次のコードは私のために機能します:

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

カメラにアクセスして写真を撮り、AndroidでImageViewを設定するには

マシュマロにはUri file = Uri.fromFile(getOutputMediaFile());を使用する必要があります。

以下のリンクを使用してパスを取得します

https://androidkennel.org/Android-camera-access-tutorial/

0
Ashwin H